diff --git a/common/rfb/CConnection.h b/common/rfb/CConnection.h index b30a997be..11a145e90 100644 --- a/common/rfb/CConnection.h +++ b/common/rfb/CConnection.h @@ -36,6 +36,18 @@ namespace rfb { class CMsgWriter; class CSecurity; + enum class MsgBoxFlags{ + M_OK = 0, + M_OKCANCEL = 1, + M_YESNO = 4, + M_ICONERROR = 0x10, + M_ICONQUESTION = 0x20, + M_ICONWARNING = 0x30, + M_ICONINFORMATION = 0x40, + M_DEFBUTTON1 = 0, + M_DEFBUTTON2 = 0x100 + }; + class CConnection : public CMsgHandler { public: @@ -111,7 +123,7 @@ namespace rfb { void serverCutText(const char* str) override; void handleClipboardCaps(uint32_t flags, - const uint32_t* lengths) override; + const uint32_t* lengths) override; void handleClipboardRequest(uint32_t flags) override; void handleClipboardPeek() override; void handleClipboardNotify(uint32_t flags) override; @@ -121,6 +133,21 @@ namespace rfb { // Methods to be overridden in a derived class + // + // \brief getUserPasswd gets the username and password. + // This might involve a dialog, getpass(), etc. + // \param secure: Indicates whether this connection is secure + // \param user: user name buffer pointer. the pointer may be nullptr, + // in which case no user name will be retrieved. + // \param password: password buffer pointer. + // + virtual void getUserPasswd(bool secure, + std::string* user, + std::string* password) = 0; + virtual bool showMsgBox(MsgBoxFlags flags, + const char *title, + const char *text) = 0; + // authSuccess() is called when authentication has succeeded. virtual void authSuccess(); diff --git a/common/rfb/CSecurity.h b/common/rfb/CSecurity.h index 549db7940..be74a1fd4 100644 --- a/common/rfb/CSecurity.h +++ b/common/rfb/CSecurity.h @@ -38,9 +38,6 @@ #ifndef __RFB_CSECURITY_H__ #define __RFB_CSECURITY_H__ -#include -#include - namespace rfb { class CConnection; class CSecurity { @@ -51,13 +48,6 @@ namespace rfb { virtual int getType() const = 0; virtual bool isSecure() const { return false; } - /* - * Use variable directly instead of dumb get/set methods. - * It MUST be set by viewer. - */ - static UserPasswdGetter *upg; - static UserMsgBox *msg; - protected: CConnection* cc; }; diff --git a/common/rfb/CSecurityDH.cxx b/common/rfb/CSecurityDH.cxx index f3b085c72..6eafbd9ca 100644 --- a/common/rfb/CSecurityDH.cxx +++ b/common/rfb/CSecurityDH.cxx @@ -108,7 +108,7 @@ void CSecurityDH::writeCredentials() std::string password; rdr::RandomStream rs; - (CSecurity::upg)->getUserPasswd(isSecure(), &username, &password); + cc->getUserPasswd(isSecure(), &username, &password); std::vector bBytes(keyLength); if (!rs.hasData(keyLength)) diff --git a/common/rfb/CSecurityMSLogonII.cxx b/common/rfb/CSecurityMSLogonII.cxx index dc8175185..f8ff36c15 100644 --- a/common/rfb/CSecurityMSLogonII.cxx +++ b/common/rfb/CSecurityMSLogonII.cxx @@ -97,7 +97,7 @@ void CSecurityMSLogonII::writeCredentials() std::string password; rdr::RandomStream rs; - (CSecurity::upg)->getUserPasswd(isSecure(), &username, &password); + cc->getUserPasswd(isSecure(), &username, &password); std::vector bBytes(8); if (!rs.hasData(8)) diff --git a/common/rfb/CSecurityPlain.cxx b/common/rfb/CSecurityPlain.cxx index d9599f9ca..1cbf62e17 100644 --- a/common/rfb/CSecurityPlain.cxx +++ b/common/rfb/CSecurityPlain.cxx @@ -23,7 +23,6 @@ #include #include -#include #include @@ -36,7 +35,7 @@ bool CSecurityPlain::processMsg() std::string username; std::string password; - (CSecurity::upg)->getUserPasswd(cc->isSecure(), &username, &password); + cc->getUserPasswd(cc->isSecure(), &username, &password); // Return the response to the server os->writeU32(username.size()); diff --git a/common/rfb/CSecurityRSAAES.cxx b/common/rfb/CSecurityRSAAES.cxx index a78739aca..cc031fe3c 100644 --- a/common/rfb/CSecurityRSAAES.cxx +++ b/common/rfb/CSecurityRSAAES.cxx @@ -38,7 +38,6 @@ #include #include #include -#include #include #include #include @@ -215,7 +214,7 @@ void CSecurityRSAAES::verifyServer() "Fingerprint: %02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x\n" "Please verify that the information is correct and press \"Yes\". " "Otherwise press \"No\"", f[0], f[1], f[2], f[3], f[4], f[5], f[6], f[7]); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, title, text.c_str())) + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, title, text.c_str())) throw Exception("server key mismatch"); } @@ -438,9 +437,9 @@ void CSecurityRSAAES::writeCredentials() std::string password; if (subtype == secTypeRA2UserPass) - (CSecurity::upg)->getUserPasswd(isSecure(), &username, &password); + cc->getUserPasswd(isSecure(), &username, &password); else - (CSecurity::upg)->getUserPasswd(isSecure(), nullptr, &password); + cc->getUserPasswd(isSecure(), nullptr, &password); if (subtype == secTypeRA2UserPass) { if (username.size() > 255) diff --git a/common/rfb/CSecurityRSAAES.h b/common/rfb/CSecurityRSAAES.h index 29bfd5750..35b0513bb 100644 --- a/common/rfb/CSecurityRSAAES.h +++ b/common/rfb/CSecurityRSAAES.h @@ -27,13 +27,11 @@ #include #include #include -#include #include #include #include namespace rfb { - class UserMsgBox; class CSecurityRSAAES : public CSecurity { public: CSecurityRSAAES(CConnection* cc, uint32_t secType, diff --git a/common/rfb/CSecurityTLS.cxx b/common/rfb/CSecurityTLS.cxx index 2a7a11794..eff215ab0 100644 --- a/common/rfb/CSecurityTLS.cxx +++ b/common/rfb/CSecurityTLS.cxx @@ -38,7 +38,6 @@ #include #include #include -#include #include #include #include @@ -442,7 +441,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Unknown certificate issuer", text.c_str())) throw AuthCancelledException(); @@ -462,8 +461,7 @@ void CSecurityTLS::checkSession() "\n" "Do you want to make an exception for this " "server?", info.data); - - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Certificate is not yet valid", text.c_str())) throw AuthCancelledException(); @@ -482,7 +480,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Expired certificate", text.c_str())) throw AuthCancelledException(); @@ -501,7 +499,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Insecure certificate algorithm", text.c_str())) throw AuthCancelledException(); @@ -526,7 +524,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", client->getServerName(), info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Certificate hostname mismatch", text.c_str())) throw AuthCancelledException(); @@ -552,7 +550,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Unexpected server certificate", text.c_str())) throw AuthCancelledException(); @@ -575,7 +573,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Unexpected server certificate", text.c_str())) throw AuthCancelledException(); @@ -596,7 +594,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Unexpected server certificate", text.c_str())) throw AuthCancelledException(); @@ -617,7 +615,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Unexpected server certificate", text.c_str())) throw AuthCancelledException(); @@ -644,7 +642,7 @@ void CSecurityTLS::checkSession() "Do you want to make an exception for this " "server?", client->getServerName(), info.data); - if (!msg->showMsgBox(UserMsgBox::M_YESNO, + if (!cc->showMsgBox(MsgBoxFlags::M_YESNO, "Unexpected server certificate", text.c_str())) throw AuthCancelledException(); diff --git a/common/rfb/CSecurityTLS.h b/common/rfb/CSecurityTLS.h index 8688b742c..848ef9bb8 100644 --- a/common/rfb/CSecurityTLS.h +++ b/common/rfb/CSecurityTLS.h @@ -28,7 +28,6 @@ #include #include -#include #include #include #include diff --git a/common/rfb/CSecurityVncAuth.cxx b/common/rfb/CSecurityVncAuth.cxx index e5f842baa..2fd83fe93 100644 --- a/common/rfb/CSecurityVncAuth.cxx +++ b/common/rfb/CSecurityVncAuth.cxx @@ -54,7 +54,7 @@ bool CSecurityVncAuth::processMsg() uint8_t challenge[vncAuthChallengeSize]; is->readBytes(challenge, vncAuthChallengeSize); std::string passwd; - (CSecurity::upg)->getUserPasswd(cc->isSecure(), nullptr, &passwd); + cc->getUserPasswd(cc->isSecure(), nullptr, &passwd); // Calculate the correct response uint8_t key[8]; diff --git a/common/rfb/SecurityClient.cxx b/common/rfb/SecurityClient.cxx index d1507eb5c..9cd3b9047 100644 --- a/common/rfb/SecurityClient.cxx +++ b/common/rfb/SecurityClient.cxx @@ -40,11 +40,6 @@ using namespace rfb; -UserPasswdGetter *CSecurity::upg = nullptr; -#if defined(HAVE_GNUTLS) || defined(HAVE_NETTLE) -UserMsgBox *CSecurity::msg = nullptr; -#endif - StringParameter SecurityClient::secTypes ("SecurityTypes", "Specify which security scheme to use (None, VncAuth, Plain" @@ -66,11 +61,6 @@ ConfViewer); CSecurity* SecurityClient::GetCSecurity(CConnection* cc, uint32_t secType) { - assert (CSecurity::upg != nullptr); /* (upg == nullptr) means bug in the viewer */ -#if defined(HAVE_GNUTLS) || defined(HAVE_NETTLE) - assert (CSecurity::msg != nullptr); -#endif - if (!IsSupported(secType)) goto bail; diff --git a/common/rfb/UserMsgBox.h b/common/rfb/UserMsgBox.h deleted file mode 100644 index 392950d12..000000000 --- a/common/rfb/UserMsgBox.h +++ /dev/null @@ -1,41 +0,0 @@ -/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. - * Copyright (C) 2010 TigerVNC Team - * - * This is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This software is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this software; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, - * USA. - */ -#ifndef __RFB_USERMSGBOX_H__ -#define __RFB_USERMSGBOX_H__ - -namespace rfb { - class UserMsgBox { - public: - enum MsgBoxFlags{ - M_OK = 0, - M_OKCANCEL = 1, - M_YESNO = 4, - M_ICONERROR = 0x10, - M_ICONQUESTION = 0x20, - M_ICONWARNING = 0x30, - M_ICONINFORMATION = 0x40, - M_DEFBUTTON1 = 0, - M_DEFBUTTON2 = 0x100 - }; - /* TODO Implement as function with variable arguments */ - virtual bool showMsgBox(int flags,const char* title, const char* text)=0; - }; -} - -#endif diff --git a/common/rfb/UserPasswdGetter.h b/common/rfb/UserPasswdGetter.h deleted file mode 100644 index db7df3965..000000000 --- a/common/rfb/UserPasswdGetter.h +++ /dev/null @@ -1,36 +0,0 @@ -/* Copyright (C) 2002-2005 RealVNC Ltd. All Rights Reserved. - * - * This is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 2 of the License, or - * (at your option) any later version. - * - * This software is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this software; if not, write to the Free Software - * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, - * USA. - */ -#ifndef __RFB_USERPASSWDGETTER_H__ -#define __RFB_USERPASSWDGETTER_H__ - -#include - -namespace rfb { - class UserPasswdGetter { - public: - // getUserPasswd gets the username and password. This might involve a - // dialog, getpass(), etc. The user buffer pointer can be null, in which - // case no user name will be retrieved. - virtual void getUserPasswd(bool secure, std::string* user, - std::string* password)=0; - - virtual ~UserPasswdGetter() {} - }; -} - -#endif diff --git a/tests/perf/decperf.cxx b/tests/perf/decperf.cxx index ef53a6866..fb2390be2 100644 --- a/tests/perf/decperf.cxx +++ b/tests/perf/decperf.cxx @@ -75,6 +75,8 @@ class CConn : public rfb::CConnection { void setColourMapEntries(int, int, uint16_t*) override; void bell() override; void serverCutText(const char*) override; + virtual void getUserPasswd(bool secure, std::string *user, std::string *password) override; + virtual bool showMsgBox(rfb::MsgBoxFlags flags, const char *title, const char *text) override; public: double cpuTime; @@ -82,6 +84,7 @@ class CConn : public rfb::CConnection { protected: rdr::FileInStream *in; DummyOutStream *out; + }; DummyOutStream::DummyOutStream() @@ -174,6 +177,15 @@ void CConn::serverCutText(const char*) { } +void CConn::getUserPasswd(bool, std::string *, std::string *) +{ +} + +bool CConn::showMsgBox(rfb::MsgBoxFlags, const char *, const char *) +{ + return true; +} + struct stats { double decodeTime; diff --git a/tests/perf/encperf.cxx b/tests/perf/encperf.cxx index ac8e9d47a..c6e2d3b7a 100644 --- a/tests/perf/encperf.cxx +++ b/tests/perf/encperf.cxx @@ -108,6 +108,8 @@ class CConn : public rfb::CConnection { void setColourMapEntries(int, int, uint16_t*) override; void bell() override; void serverCutText(const char*) override; + virtual void getUserPasswd(bool secure, std::string *user, std::string *password) override; + virtual bool showMsgBox(rfb::MsgBoxFlags flags, const char *title, const char *text) override; public: double decodeTime; @@ -279,6 +281,15 @@ void CConn::serverCutText(const char*) { } +void CConn::getUserPasswd(bool, std::string *, std::string *) +{ +} + +bool CConn::showMsgBox(rfb::MsgBoxFlags, const char *, const char *) +{ + return true; +} + Manager::Manager(class rfb::SConnection *conn_) : EncodeManager(conn_) { diff --git a/vncviewer/CConn.cxx b/vncviewer/CConn.cxx index e1c759620..a23cfbd6a 100644 --- a/vncviewer/CConn.cxx +++ b/vncviewer/CConn.cxx @@ -275,7 +275,7 @@ void CConn::socketEvent(FL_SOCKET fd, void *data) vlog.info("%s", e.str()); disconnect(); } catch (rfb::AuthFailureException& e) { - reset_password_data(); + cc->resetPassword(); vlog.error(_("Authentication failed: %s"), e.str()); abort_connection(_("Failed to authenticate with the server. Reason " "given by the server:\n\n%s"), e.str()); @@ -606,3 +606,18 @@ void CConn::handleUpdateTimeout(void *data) Fl::repeat_timeout(1.0, handleUpdateTimeout, data); } + +bool CConn::showMsgBox(MsgBoxFlags flags, const char *title, const char *text) +{ + return dlg.showMsgBox(flags, title, text); +} + +void CConn::getUserPasswd(bool secure, std::string *user, std::string *password) +{ + dlg.getUserPasswd(secure, user, password); +} + +void CConn::resetPassword() +{ + dlg.resetPassword(); +} \ No newline at end of file diff --git a/vncviewer/CConn.h b/vncviewer/CConn.h index fcaf49f6e..f18aeb169 100644 --- a/vncviewer/CConn.h +++ b/vncviewer/CConn.h @@ -24,6 +24,7 @@ #include #include +#include "UserDialog.h" namespace network { class Socket; } @@ -44,6 +45,11 @@ class CConn : public rfb::CConnection // Callback when socket is ready (or broken) static void socketEvent(FL_SOCKET fd, void *data); + // UserMsgBox interface + virtual bool showMsgBox(rfb::MsgBoxFlags flags, const char *title, const char *text) override; + // UserPasswdGetter interface + virtual void getUserPasswd(bool secure, std::string *user, std::string *password) override; + // CConnection callback methods void initDone() override; @@ -76,6 +82,8 @@ class CConn : public rfb::CConnection void handleClipboardAnnounce(bool available) override; void handleClipboardData(const char* data) override; + void resetPassword(); + private: void resizeFramebuffer() override; @@ -105,6 +113,8 @@ class CConn : public rfb::CConnection struct timeval updateStartTime; size_t updateStartPos; unsigned long long bpsEstimate; + + UserDialog dlg; }; #endif diff --git a/vncviewer/UserDialog.cxx b/vncviewer/UserDialog.cxx index a37678576..18b30582b 100644 --- a/vncviewer/UserDialog.cxx +++ b/vncviewer/UserDialog.cxx @@ -248,7 +248,7 @@ void UserDialog::getUserPasswd(bool secure_, std::string* user, throw rfb::AuthCancelledException(); } -bool UserDialog::showMsgBox(int flags, const char* title, const char* text) +bool UserDialog::showMsgBox(MsgBoxFlags flags, const char* title, const char* text) { char buffer[1024]; @@ -260,15 +260,15 @@ bool UserDialog::showMsgBox(int flags, const char* title, const char* text) fl_message_title(title); - switch (flags & 0xf) { - case M_OKCANCEL: + switch ((MsgBoxFlags)((int)flags & 0xf)) { + case MsgBoxFlags::M_OKCANCEL: return fl_choice("%s", nullptr, fl_ok, fl_cancel, buffer) == 1; - case M_YESNO: + case MsgBoxFlags::M_YESNO: return fl_choice("%s", nullptr, fl_yes, fl_no, buffer) == 1; - case M_OK: + case MsgBoxFlags::M_OK: default: - if (((flags & 0xf0) == M_ICONERROR) || - ((flags & 0xf0) == M_ICONWARNING)) + if ((((int)flags & 0xf0) == (int)MsgBoxFlags::M_ICONERROR) || + (((int)flags & 0xf0) == (int)MsgBoxFlags::M_ICONWARNING)) fl_alert("%s", buffer); else fl_message("%s", buffer); diff --git a/vncviewer/UserDialog.h b/vncviewer/UserDialog.h index aa50127e7..c16923c12 100644 --- a/vncviewer/UserDialog.h +++ b/vncviewer/UserDialog.h @@ -19,11 +19,9 @@ #ifndef __USERDIALOG_H__ #define __USERDIALOG_H__ -#include -#include +#include -class UserDialog : public rfb::UserPasswdGetter, - public rfb::UserMsgBox +class UserDialog { public: UserDialog(); @@ -32,15 +30,14 @@ class UserDialog : public rfb::UserPasswdGetter, // UserPasswdGetter callbacks void getUserPasswd(bool secure, std::string* user, - std::string* password) override; + std::string* password); // UserMsgBox callbacks - - bool showMsgBox(int flags, const char* title, const char* text) override; + bool showMsgBox(rfb::MsgBoxFlags flags, const char* title, const char* text); void resetPassword(); - private: +private: std::string savedUsername; std::string savedPassword; }; diff --git a/vncviewer/vncviewer.cxx b/vncviewer/vncviewer.cxx index 27946f3fc..26782815f 100644 --- a/vncviewer/vncviewer.cxx +++ b/vncviewer/vncviewer.cxx @@ -93,8 +93,6 @@ static bool exitMainloop = false; static char *exitError = nullptr; static bool fatalError = false; -static UserDialog dlg; - static const char *about_text() { static char buffer[1024]; @@ -171,11 +169,6 @@ bool should_disconnect() return exitMainloop; } -void reset_password_data() -{ - dlg.resetPassword(); -} - void about_vncviewer() { fl_message_title(_("About TigerVNC Viewer")); @@ -748,11 +741,6 @@ int main(int argc, char** argv) vlog.error(_("Could not create VNC state directory: %s"), strerror(errno)); } - CSecurity::upg = &dlg; -#if defined(HAVE_GNUTLS) || defined(HAVE_NETTLE) - CSecurity::msg = &dlg; -#endif - Socket *sock = nullptr; #ifndef WIN32 diff --git a/vncviewer/vncviewer.h b/vncviewer/vncviewer.h index c864805cd..f39a57761 100644 --- a/vncviewer/vncviewer.h +++ b/vncviewer/vncviewer.h @@ -33,7 +33,6 @@ void abort_connection_with_unexpected_error(const rdr::Exception &); void disconnect(); bool should_disconnect(); -void reset_password_data(); void about_vncviewer();