Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1152347
  • 博文数量: 222
  • 博客积分: 5262
  • 博客等级: 大校
  • 技术积分: 3028
  • 用 户 组: 普通用户
  • 注册时间: 2009-11-22 19:10
文章分类

全部博文(222)

文章存档

2012年(2)

2011年(192)

2010年(28)

分类: 嵌入式

2011-07-14 19:58:35

    由于测试vnc反向连接出错,所以这几天一直在调试vnc的反向连接问题。
    通过初步调试得知是在协议的server端连接到viewer端的版本信息验证错误,故追查之。
    要让vncviewer端支持反向连接,可在命令行加参数,也可以修还源码的vncviewer.cxx文件中的linstenmode为ture。
    相对于viewer端,server的情况比较复杂一点,跟踪了好几天终于找到地方了,关键点在vncconndialog.cpp文件,下面把重要的源代码部分贴出来。
   
// Callback function - handles messages sent to the dialog box

BOOL CALLBACK vncConnDialog::vncConnDlgProc(HWND hwnd,
                                            UINT uMsg,
                                            WPARAM wParam,
                                            LPARAM lParam) {
    // This is a static method, so we don't know which instantiation we're
    // dealing with. But we can get a pseudo-this from the parameter to
    // WM_INITDIALOG, which we therafter store with the window and retrieve
    // as follows:
     vncConnDialog *_this = helper::SafeGetWindowUserData(hwnd);

    switch (uMsg) {
    case WM_PAINT:
        {
            //adzm 2009-06-20
            if (GetUpdateRect(hwnd, NULL, TRUE)) {
                PAINTSTRUCT ps;
                HDC hdc = BeginPaint(hwnd, &ps);

                {
                    RECT rcIcon;
                    rcIcon.top = 0;
                    rcIcon.left = 0;
                    rcIcon.bottom = 48;
                    rcIcon.right = 48;
                   
                    RECT rcClient;
                    if (GetClientRect(hwnd, &rcClient)) {
                        int nDifference = (rcClient.bottom - rcIcon.bottom) / 2;
                        if (nDifference > 0) {
                            rcIcon.top += nDifference;
                            rcIcon.bottom += nDifference;
                        }
                    }

                    RECT rcLabel;
                    HWND hwndLabel = GetDlgItem(hwnd, IDC_CONNECTION_NUMBER_STATIC);
                    if (GetWindowRect(hwndLabel, &rcLabel)) {
                        LPRECT lprcLabel = &rcLabel;
                        ScreenToClient(hwnd, (LPPOINT)lprcLabel);
                        ScreenToClient(hwnd, ((LPPOINT)lprcLabel)+1);

                        int nAdjustment = (rcLabel.left - rcIcon.right) / 2;

                        rcIcon.left += nAdjustment;
                        rcIcon.right += nAdjustment;
                    }

                    RECT rcIntersect;

                    if (IntersectRect(&rcIntersect, &rcIcon, &(ps.rcPaint))) {
                        if (_this->m_hicon == NULL) {
                            _this->m_hicon = (HICON)LoadImage(hInstResDLL, MAKEINTRESOURCE(IDI_WINVNC), IMAGE_ICON,
                               48, 48, 0);
                        }
                        if (_this->m_hicon) {
                            HBRUSH hbr = (HBRUSH)SendMessage(hwnd, WM_CTLCOLORDLG, (WPARAM)hdc, (WPARAM)hwnd);
                            DrawIconEx(hdc, rcIcon.left, rcIcon.top, _this->m_hicon, 48, 48, 0, hbr, DI_NORMAL);
                        }
                    }
                }

                EndPaint(hwnd, &ps);
            }

            return 0;
        }

        // Dialog has just been created
    case WM_INITDIALOG:
        {
            // Save the lParam into our user data so that subsequent calls have
            // access to the parent C++ object
            helper::SafeSetWindowUserData(hwnd, lParam);

            vncConnDialog *_this = (vncConnDialog *) lParam;

            //adzm 2009-06-20
            if (g_szRepeaterHost) {
                SetDlgItemText(hwnd, IDC_HOSTNAME_EDIT, g_szRepeaterHost);

                //adzm 2009-06-20
                if (SPECIAL_SC_PROMPT) {
                    HWND hwndChild = GetDlgItem(hwnd, IDC_HOSTNAME_EDIT);
                    if (hwndChild) {
                        ShowWindow(hwndChild, SW_HIDE);
                    }
                    hwndChild = GetDlgItem(hwnd, IDC_HOSTNAME_STATIC);
                    if (hwndChild) {
                        ShowWindow(hwndChild, SW_HIDE);
                    }

                    HWND hwndEdit = GetDlgItem(hwnd, IDC_IDCODE);
                    HWND hwndLabel = GetDlgItem(hwnd, IDC_CONNECTION_NUMBER_STATIC);

                    RECT rcEdit;
                    RECT rcLabel;
                    GetWindowRect(hwndEdit, &rcEdit);
                    GetWindowRect(hwndLabel, &rcLabel);

                    LPRECT lprcEdit = &rcEdit;
                    LPRECT lprcLabel = &rcLabel;

                    ScreenToClient(hwnd, (LPPOINT)lprcEdit);
                    ScreenToClient(hwnd, ((LPPOINT)lprcEdit)+1);
                   
                    ScreenToClient(hwnd, (LPPOINT)lprcLabel);
                    ScreenToClient(hwnd, ((LPPOINT)lprcLabel)+1);

                    RECT rcClient;
                    GetClientRect(hwnd, &rcClient);

                    long nTotalHeight = rcEdit.bottom - rcLabel.top;

                    long nAdjustedTop = (rcClient.bottom - nTotalHeight) / 2;

                    long nAdjustment = nAdjustedTop - rcLabel.top;

                    MoveWindow(hwndLabel, rcLabel.left, rcLabel.top + nAdjustment, rcLabel.right - rcLabel.left, rcLabel.bottom - rcLabel.top, TRUE);
                    MoveWindow(hwndEdit, rcEdit.left, rcEdit.top + nAdjustment, rcEdit.right - rcEdit.left, rcEdit.bottom - rcEdit.top, TRUE);

                    HWND hwndCaption = GetDlgItem(hwnd, IDC_CAPTION_STATIC);
                    HFONT hFont = (HFONT)SendMessage(hwndCaption, WM_GETFONT, 0, 0);
                    if (hFont) {
                        LOGFONT lf;
                        if (GetObject(hFont, sizeof(LOGFONT), &lf)) {
                            lf.lfWidth = 0;
                            lf.lfHeight = (lf.lfHeight * 6) / 4;

                            _this->m_hfont = CreateFontIndirect(&lf);

                            if (_this->m_hfont) {
                                SendMessage(hwndCaption, WM_SETFONT, (WPARAM)_this->m_hfont, (LPARAM)TRUE);
                            }
                        }
                    }
                    SetWindowText(hwndCaption, "Connect to Technical Support");
                    ShowWindow(hwndCaption, SW_SHOWNA);
                }

                SetFocus(GetDlgItem(hwnd, IDC_IDCODE));
            } else {           
                // Make the text entry box active
                SetFocus(GetDlgItem(hwnd, IDC_HOSTNAME_EDIT));
            }
           
            SetForegroundWindow(hwnd);
           
            // Return success!
            return TRUE;
        }

        // Dialog has just received a command
    case WM_COMMAND:
        switch (LOWORD(wParam))
        {
            // User clicked OK or pressed return
        case IDOK:
        {
            // sf@2002 - host:num & host::num analyse.
            // Compatible with both RealVNC and TightVNC methods
            char hostname[_MAX_PATH];
            char actualhostname[_MAX_PATH];
            char idcode[_MAX_PATH];
            char *portp;
            int port;
            bool id;

            // Get the hostname of the VNCviewer
            GetDlgItemText(hwnd, IDC_HOSTNAME_EDIT, hostname, _MAX_PATH);
            GetDlgItemText(hwnd, IDC_IDCODE, idcode, _MAX_PATH);
            if (strcmp(idcode,"")==NULL) id=false;
            else id=true;

            strcpy(actualhostname, hostname);
           
            //adzm 2010-02-15 - Multiple repeaters chosen by modulo of ID

            char finalidcode[_MAX_PATH];

            //adzm 2009-06-20
            if (id) {
                size_t i = 0;

                for (i = 0; i < strlen(idcode); i++)
                {
                    finalidcode[i] = toupper(idcode[i]);
                }
                finalidcode[i] = 0;

                if (0 != strncmp("ID:", idcode, 3)) {
                    strcpy(finalidcode, "ID:");
                   
                    for (i = 0; i < strlen(idcode); i++)
                    {
                        finalidcode[i+3] = toupper(idcode[i]);
                    }
                    finalidcode[i+3] = 0;
                }

               
                //adzm 2010-02-15 - At this point, finalidcode is of the form "ID:#####"
                int numericId = atoi(finalidcode + 3);

                int numberOfHosts = 1;
                for (i = 0; i < strlen(hostname); i++) {
                    if (hostname[i] == ';') {
                        numberOfHosts++;
                    }
                }

                if (numberOfHosts <= 1) {
                    // then hostname == actualhostname
                } else {
                    int modulo = numericId % numberOfHosts;

                    char* szToken = strtok(hostname, ";");
                    while (szToken) {
                        if (modulo == 0) {
                            strcpy(actualhostname, szToken);
                            break;
                        }

                        modulo--;
                        szToken = strtok(NULL, ";");
                    }
                }
            }

            // Calculate the Display and Port offset.
            port = INCOMING_PORT_OFFSET;
            portp = strchr(actualhostname, ':');
            if (portp)
            {
                *portp++ = '\0';
                if (*portp == ':') // Tight127 method
                {
                    port = atoi(++portp);        // Port number after "::"
                }
                else // RealVNC method
                {
                    if (atoi(portp) < 100)        // If < 100 after ":" -> display number
                        port += atoi(portp);
                    else
                        port = atoi(portp);        // If > 100 after ":" -> Port number
                }
            }


           
            // Attempt to create a new socket
            VSocket *tmpsock;
            tmpsock = new VSocket;
            if (!tmpsock) {
                return TRUE;
            }
           
            // Connect out to the specified host on the VNCviewer listen port
            // To be really good, we should allow a display number here but
            // for now we'll just assume we're connecting to display zero
            tmpsock->Create();
            if (tmpsock->Connect(actualhostname, port))
            {
                     id=false;                   //debug by qz.这个地方我们要绕过去,不然就连接不上客户端了。
                if (id)
                    {                           
                       

                        tmpsock->Send(finalidcode,250);
                        tmpsock->SetTimeout(0);
/*                        if (strncmp(hostname,"ID",2)!=0)
                        {
                        while (true)
                        {
                            char result[1];
                            tmpsock->Read(result,1);
                            if (strcmp(result,"2")==0)
                                break;
                            tmpsock->Send("1",1);
                        }
                        }*/
                       
                        // adzm 2009-07-05 - repeater IDs
                        // Add the new client to this server
                        // adzm 2009-08-02
                        _this->m_server->AddClient(tmpsock, TRUE, TRUE, 0, NULL, finalidcode, actualhostname, port);
                    } else {
                        // Add the new client to this server
                        // adzm 2009-08-02
                        _this->m_server->AddClient(tmpsock, TRUE, TRUE, 0, NULL, NULL, actualhostname, port);               
                    }
                // And close the dialog
                EndDialog(hwnd, TRUE);
            }
            else
            {
                // Print up an error message
                MessageBox(NULL,
                    sz_ID_FAILED_CONNECT_LISTING_VIEW,
                    sz_ID_OUTGOING_CONNECTION,
                    MB_OK | MB_ICONEXCLAMATION );
                delete tmpsock;
            }
            return TRUE;
        }
       
            // Cancel the dialog
        case IDCANCEL:
            EndDialog(hwnd, FALSE);
            return TRUE;
        };

        break;

    case WM_DESTROY:
        EndDialog(hwnd, FALSE);
        if (_this->m_hicon != NULL) {
            DestroyIcon(_this->m_hicon);
            _this->m_hicon = NULL;
        }
        if (_this->m_hfont != NULL) {
            DeleteObject(_this->m_hfont);
            _this->m_hfont = NULL;
        }
        return TRUE;
    }
    return 0;
}


阅读(7415) | 评论(0) | 转发(1) |
给主人留下些什么吧!~~