Changeset 36258 in vbox for trunk/src/VBox/Additions
- Timestamp:
- Mar 11, 2011 9:53:23 AM (14 years ago)
- svn:sync-xref-src-repo-rev:
- 70498
- Location:
- trunk/src/VBox/Additions/WINNT/VBoxGINA
- Files:
-
- 2 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Additions/WINNT/VBoxGINA/Dialog.cpp
r30852 r36258 6 6 /* 7 7 * 8 * Copyright (C) 2006-201 0Oracle Corporation8 * Copyright (C) 2006-2011 Oracle Corporation 9 9 * 10 10 * This file is part of VirtualBox Open Source Edition (OSE), as … … 19 19 #include <windows.h> 20 20 #include <stdio.h> /* Needed for swprintf() */ 21 21 22 #include "Dialog.h" 22 23 #include "WinWlx.h" … … 24 25 #include "VBoxGINA.h" 25 26 26 // 27 // MSGINA dialog box IDs. 28 // 27 28 /* 29 * Dialog IDs for legacy Windows OSes (e.g. NT 4.0). 30 */ 29 31 #define IDD_WLXDIAPLAYSASNOTICE_DIALOG 1400 30 32 #define IDD_WLXLOGGEDOUTSAS_DIALOG 1450 31 /* the Windows 2000 ID */32 #define IDD_WLXLOGGEDOUTSAS_DIALOG2 1500 33 /** Change password dialog: To change the current 34 * account password. */ 33 35 #define IDD_CHANGE_PASSWORD_DIALOG 1550 34 36 #define IDD_WLXLOGGEDONSAS_DIALOG 1650 37 /** Security dialog: To lock the workstation, log off 38 * change password, ... */ 39 #define IDD_SECURITY_DIALOG 1800 40 /** Locked dialog: To unlock the currently lockted 41 * workstation. */ 35 42 #define IDD_WLXWKSTALOCKEDSAS_DIALOG 1850 36 37 // 38 // MSGINA control IDs 39 // 43 /** Shutdown dialog: To either restart, logoff current 44 * user or shutdown the workstation. */ 45 #define IDD_SHUTDOWN_DIALOG 2200 46 /** Logoff dialog: "Do you really want to logoff?". */ 47 #define IDD_LOGOFF_DIALOG 2250 48 49 50 /* 51 * Dialog IDs for Windows 2000 and up. 52 */ 53 #define IDD_WLXLOGGEDOUTSAS_DIALOG2 1500 54 /** Change password dialog: To change the current 55 * account password. */ 56 #define IDD_CHANGE_PASSWORD_DIALOG2 1700 57 /** Locked dialog: To unlock the currently lockted 58 * workstation. */ 59 #define IDD_WLXWKSTALOCKEDSAS_DIALOG2 1950 60 61 62 /* 63 * Control IDs. 64 */ 40 65 #define IDC_WLXLOGGEDOUTSAS_USERNAME 1453 41 66 #define IDC_WLXLOGGEDOUTSAS_USERNAME2 1502 … … 44 69 #define IDC_WLXLOGGEDOUTSAS_DOMAIN 1455 45 70 #define IDC_WLXLOGGEDOUTSAS_DOMAIN2 1504 46 #define IDC_WLXWKSTALOCKEDSAS_DOMAIN 1856 71 72 #define IDC_WKSTALOCKED_USERNAME 1953 73 #define IDC_WKSTALOCKED_PASSWORD 1954 74 #define IDC_WKSTALOCKEd_DOMAIN 1856 75 #define IDC_WKSTALOCKED_DOMAIN2 1956 76 77 78 /* 79 * Own IDs. 80 */ 81 #define IDT_BASE WM_USER + 1100 /* Timer ID base. */ 82 #define IDT_LOGGEDONDLG_POLL IDT_BASE + 1 83 #define IDT_LOCKEDDLG_POLL IDT_BASE + 2 47 84 48 85 static DLGPROC g_pfnWlxLoggedOutSASDlgProc = NULL; 86 static DLGPROC g_pfnWlxLockedSASDlgProc = NULL; 49 87 50 88 static PWLX_DIALOG_BOX_PARAM g_pfnWlxDialogBoxParam = NULL; … … 106 144 // Redirected WlxLoggedOutSASDlgProc(). 107 145 // 108 109 #define CREDPOLL_TIMERID 0x1243110 111 146 BOOL credentialsToUI(HWND hwndUserId, HWND hwndPassword, HWND hwndDomain) 112 147 { … … 183 218 { 184 219 BOOL bResult; 185 static HWND hwndUserId, hwndPassword, hwndDomain = 0; 186 static UINT_PTR timer = 0; 220 static HWND s_hwndUserId, s_hwndPassword, s_hwndDomain = 0; 187 221 188 222 /*Log(("VBoxGINA::MyWlxLoggedOutSASDlgProc\n"));*/ … … 203 237 204 238 /* get the entry fields */ 205 hwndUserId = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_USERNAME);206 if (! hwndUserId)207 hwndUserId = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_USERNAME2);208 hwndPassword = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_PASSWORD);209 if (! hwndPassword)210 hwndPassword = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_PASSWORD2);211 hwndDomain = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_DOMAIN);212 if (! hwndDomain)213 hwndDomain = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_DOMAIN2);239 s_hwndUserId = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_USERNAME); 240 if (!s_hwndUserId) 241 s_hwndUserId = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_USERNAME2); 242 s_hwndPassword = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_PASSWORD); 243 if (!s_hwndPassword) 244 s_hwndPassword = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_PASSWORD2); 245 s_hwndDomain = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_DOMAIN); 246 if (!s_hwndDomain) 247 s_hwndDomain = GetDlgItem(hwndDlg, IDC_WLXLOGGEDOUTSAS_DOMAIN2); 214 248 215 249 Log(("VBoxGINA::MyWlxLoggedOutSASDlgProc: hwndUserId: %x, hwndPassword: %d, hwndDomain: %d\n", 216 hwndUserId, hwndPassword,hwndDomain));250 s_hwndUserId, s_hwndPassword, s_hwndDomain)); 217 251 218 252 /* terminate the credentials poller thread, it's done is job */ … … 225 259 { 226 260 /* fill in credentials to appropriate UI elements */ 227 credentialsToUI( hwndUserId, hwndPassword,hwndDomain);261 credentialsToUI(s_hwndUserId, s_hwndPassword, s_hwndDomain); 228 262 229 263 /* we got the credentials, null them out */ … … 241 275 * Create a timer and poll for them. 242 276 */ 243 timer = SetTimer(hwndDlg, CREDPOLL_TIMERID, 200, NULL); 244 if (!timer) 245 { 246 Log(("VBoxGINA::MyWlxLoggedOutSASDlgProc: failed creating timer! last error: %s\n", 277 UINT_PTR uTimer = SetTimer(hwndDlg, IDT_LOGGEDONDLG_POLL, 200, NULL); 278 if (!uTimer) 279 Log(("VBoxGINA::MyWlxLoggedOutSASDlgProc: failed creating timer! Last error: %ld\n", 247 280 GetLastError())); 248 }249 281 } 250 282 break; … … 254 286 { 255 287 /* is it our credentials poller timer? */ 256 if (wParam == CREDPOLL_TIMERID)288 if (wParam == IDT_LOGGEDONDLG_POLL) 257 289 { 258 290 if (credentialsAvailable()) … … 261 293 { 262 294 /* fill in credentials to appropriate UI elements */ 263 credentialsToUI( hwndUserId, hwndPassword,hwndDomain);295 credentialsToUI(s_hwndUserId, s_hwndPassword, s_hwndDomain); 264 296 265 297 /* we got the credentials, null them out */ … … 271 303 272 304 /* we don't need the timer any longer */ 273 /** @todo will we leak the timer when logging in manually? Should we kill it on WM_CLOSE? */ 274 KillTimer(hwndDlg, CREDPOLL_TIMERID); 305 KillTimer(hwndDlg, IDT_LOGGEDONDLG_POLL); 275 306 } 276 307 } … … 278 309 break; 279 310 } 311 312 case WM_DESTROY: 313 KillTimer(hwndDlg, IDT_LOGGEDONDLG_POLL); 314 break; 315 } 316 return bResult; 317 } 318 319 320 INT_PTR CALLBACK MyWlxLockedSASDlgProc(HWND hwndDlg, // handle to dialog box 321 UINT uMsg, // message 322 WPARAM wParam, // first message parameter 323 LPARAM lParam) // second message parameter 324 { 325 BOOL bResult; 326 static HWND s_hwndPassword = 0; 327 328 /*Log(("VBoxGINA::MyWlxLockedSASDlgProc\n"));*/ 329 330 // 331 // Pass on to MSGINA first. 332 // 333 bResult = g_pfnWlxLockedSASDlgProc(hwndDlg, uMsg, wParam, lParam); 334 335 // 336 // We are only interested in the WM_INITDIALOG message. 337 // 338 switch (uMsg) 339 { 340 case WM_INITDIALOG: 341 { 342 Log(("VBoxGINA::MyWlxLockedSASDlgProc: got WM_INITDIALOG\n")); 343 344 /* get the entry fields */ 345 s_hwndPassword = GetDlgItem(hwndDlg, IDC_WKSTALOCKED_PASSWORD); 346 Log(("VBoxGINA::MyWlxLockedSASDlgProc: hwndPassword: %d\n", s_hwndPassword)); 347 348 /* terminate the credentials poller thread, it's done is job */ 349 credentialsPollerTerminate(); 350 351 if (credentialsAvailable()) 352 { 353 /* query the credentials from VBox */ 354 if (credentialsRetrieve()) 355 { 356 /* fill in credentials to appropriate UI elements */ 357 credentialsToUI(NULL /* User ID */, s_hwndPassword, NULL /* Domain */); 358 359 /* we got the credentials, null them out */ 360 credentialsReset(); 361 362 /* confirm the logon dialog, simulating the user pressing "OK" */ 363 WPARAM wParam = MAKEWPARAM(IDOK, BN_CLICKED); 364 PostMessage(hwndDlg, WM_COMMAND, wParam, 0); 365 } 366 } 367 else 368 { 369 /* 370 * The dialog is there but we don't have any credentials. 371 * Create a timer and poll for them. 372 */ 373 UINT_PTR uTimer = SetTimer(hwndDlg, IDT_LOCKEDDLG_POLL, 200, NULL); 374 if (!uTimer) 375 Log(("VBoxGINA::MyWlxLockedSASDlgProc: failed creating timer! Last error: %ld\n", 376 GetLastError())); 377 } 378 break; 379 } 380 381 case WM_TIMER: 382 { 383 /* is it our credentials poller timer? */ 384 if (wParam == IDT_LOCKEDDLG_POLL) 385 { 386 if (credentialsAvailable()) 387 { 388 if (credentialsRetrieve()) 389 { 390 /* fill in credentials to appropriate UI elements */ 391 credentialsToUI(NULL /* User ID */, s_hwndPassword, NULL /* Domain */); 392 393 /* we got the credentials, null them out */ 394 credentialsReset(); 395 396 /* confirm the logon dialog, simulating the user pressing "OK" */ 397 WPARAM wParam = MAKEWPARAM(IDOK, BN_CLICKED); 398 PostMessage(hwndDlg, WM_COMMAND, wParam, 0); 399 400 /* we don't need the timer any longer */ 401 KillTimer(hwndDlg, IDT_LOCKEDDLG_POLL); 402 } 403 } 404 } 405 break; 406 } 407 408 case WM_DESTROY: 409 KillTimer(hwndDlg, IDT_LOCKEDDLG_POLL); 410 break; 280 411 } 281 412 return bResult; … … 290 421 LPARAM dwInitParam) 291 422 { 292 Log(("VBoxGINA::MyWlxDialogBoxParam: lpszTemplate = %d\n", lpszTemplate)); 293 294 // 295 // We only know MSGINA dialogs by identifiers. 296 // 297 if (!HIWORD((int)(void*)lpszTemplate)) 298 { 299 // 300 // Hook appropriate dialog boxes as necessary. 301 // 302 switch ((DWORD) lpszTemplate) 303 { 304 case IDD_WLXLOGGEDOUTSAS_DIALOG: 305 case IDD_WLXLOGGEDOUTSAS_DIALOG2: 306 { 307 Log(("VBoxGINA::MyWlxDialogBoxParam: returning hooked logged out dialog\n")); 308 g_pfnWlxLoggedOutSASDlgProc = dlgprc; 309 return g_pfnWlxDialogBoxParam(hWlx, hInst, lpszTemplate, hwndOwner, 310 MyWlxLoggedOutSASDlgProc, dwInitParam); 311 } 312 } 313 } 314 315 // 316 // The rest will not be redirected. 317 // 318 return g_pfnWlxDialogBoxParam(hWlx, hInst, lpszTemplate, 319 hwndOwner, dlgprc, dwInitParam); 423 Log(("VBoxGINA::MyWlxDialogBoxParam: lpszTemplate = %ls\n", lpszTemplate)); 424 425 // 426 // We only know MSGINA dialogs by identifiers. 427 // 428 if (!HIWORD((int)(void*)lpszTemplate)) 429 { 430 // 431 // Hook appropriate dialog boxes as necessary. 432 // 433 switch ((DWORD) lpszTemplate) 434 { 435 case IDD_WLXLOGGEDOUTSAS_DIALOG: /* Windows NT 4.0. */ 436 case IDD_WLXLOGGEDOUTSAS_DIALOG2: /* Windows 2000 and up. */ 437 { 438 Log(("VBoxGINA::MyWlxDialogBoxParam: returning hooked LOGGED OUT dialog\n")); 439 g_pfnWlxLoggedOutSASDlgProc = dlgprc; 440 return g_pfnWlxDialogBoxParam(hWlx, hInst, lpszTemplate, hwndOwner, 441 MyWlxLoggedOutSASDlgProc, dwInitParam); 442 } 443 444 case IDD_WLXWKSTALOCKEDSAS_DIALOG: /* Windows NT 4.0. */ 445 case IDD_WLXWKSTALOCKEDSAS_DIALOG2: /* Windows 2000 and up. */ 446 { 447 Log(("VBoxGINA::MyWlxDialogBoxParam: returning hooked LOCKED dialog\n")); 448 g_pfnWlxLockedSASDlgProc = dlgprc; 449 return g_pfnWlxDialogBoxParam(hWlx, hInst, lpszTemplate, hwndOwner, 450 MyWlxLockedSASDlgProc, dwInitParam); 451 } 452 453 /** @todo Add other hooking stuff here. */ 454 455 default: 456 { 457 char szBuf[1024]; 458 sprintf(szBuf, "VBoxGINA::MyWlxDialogBoxParam: dialog %ld not handled\n", (DWORD)lpszTemplate); 459 Log((szBuf)); 460 break; 461 } 462 } 463 } 464 465 // 466 // The rest will not be redirected. 467 // 468 return g_pfnWlxDialogBoxParam(hWlx, hInst, lpszTemplate, 469 hwndOwner, dlgprc, dwInitParam); 320 470 } 321 471 -
trunk/src/VBox/Additions/WINNT/VBoxGINA/VBoxGINA.cpp
r36012 r36258 3 3 * VBoxGINA -- Windows Logon DLL for VirtualBox 4 4 * 5 * Copyright (C) 2006-20 07Oracle Corporation5 * Copyright (C) 2006-2011 Oracle Corporation 6 6 * 7 7 * This file is part of VirtualBox Open Source Edition (OSE), as … … 24 24 25 25 /* 26 * Global variables 27 */ 28 29 30 /** DLL instance handle */ 26 * Global variables. 27 */ 28 29 /** DLL instance handle. */ 31 30 HINSTANCE hDllInstance; 32 31 33 /** Version of Winlogon */32 /** Version of Winlogon. */ 34 33 DWORD wlxVersion; 35 34 36 /** Handle to Winlogon service */35 /** Handle to Winlogon service. */ 37 36 HANDLE hGinaWlx; 38 /** Winlog function dispatch table */37 /** Winlog function dispatch table. */ 39 38 PWLX_DISPATCH_VERSION_1_1 pWlxFuncs; 40 39 41 40 /** 42 * Function pointers to MSGINA entry points 41 * Function pointers to MSGINA entry points. 43 42 */ 44 43 PGWLXNEGOTIATE GWlxNegotiate; … … 54 53 PGWLXLOGOFF GWlxLogoff; 55 54 PGWLXSHUTDOWN GWlxShutdown; 56 /* GINA 1.1 */55 /* GINA 1.1. */ 57 56 PGWLXSTARTAPPLICATION GWlxStartApplication; 58 57 PGWLXSCREENSAVERNOTIFY GWlxScreenSaverNotify; 59 /* GINA 1.3 */58 /* GINA 1.3. */ 60 59 PGWLXNETWORKPROVIDERLOAD GWlxNetworkProviderLoad; 61 60 PGWLXDISPLAYSTATUSMESSAGE GWlxDisplayStatusMessage; 62 61 PGWLXGETSTATUSMESSAGE GWlxGetStatusMessage; 63 62 PGWLXREMOVESTATUSMESSAGE GWlxRemoveStatusMessage; 64 /* GINA 1.4 */63 /* GINA 1.4. */ 65 64 PGWLXGETCONSOLESWITCHCREDENTIALS GWlxGetConsoleSwitchCredentials; 66 65 PGWLXRECONNECTNOTIFY GWlxReconnectNotify; 67 66 PGWLXDISCONNECTNOTIFY GWlxDisconnectNotify; 68 69 67 70 68 … … 82 80 RTR3Init(); 83 81 VbglR3Init(); 84 LogRel(("VBoxG ina: DLL loaded.\n"));82 LogRel(("VBoxGINA: DLL loaded.\n")); 85 83 86 84 DisableThreadLibraryCalls(hInstance); … … 91 89 case DLL_PROCESS_DETACH: 92 90 { 93 LogRel(("VBoxG ina: DLL unloaded.\n"));91 LogRel(("VBoxGINA: DLL unloaded.\n")); 94 92 VbglR3Term(); 95 93 /// @todo RTR3Term(); … … 102 100 return TRUE; 103 101 } 102 104 103 105 104 BOOL WINAPI WlxNegotiate(DWORD dwWinlogonVersion, … … 272 271 /* start the credentials poller thread */ 273 272 credentialsPollerCreate(); 274 /* forward call to MSGINA*/273 /* Forward call to MSGINA. */ 275 274 GWlxDisplaySASNotice(pWlxContext); 276 275 } … … 286 285 /* when performing a direct logon without C-A-D, our poller might not be running */ 287 286 if (!credentialsAvailable()) 288 {289 287 credentialsPollerCreate(); 290 }291 288 292 289 int iRet; … … 304 301 // pMprNotifyInfo->pszPassword 305 302 // pMprNotifyInfo->pszOldPassword 306 307 303 } 308 304 … … 316 312 Log(("VBoxGINA::WlxActivateUserShell\n")); 317 313 318 /* forward call to MSGINA*/314 /* Forward call to MSGINA. */ 319 315 return GWlxActivateUserShell(pWlxContext, pszDesktopName, pszMprLogonScript, pEnvironment); 320 316 } … … 339 335 { 340 336 Log(("VBoxGINA::WlxDisplayLockedNotice\n")); 341 /* forward call to MSGINA*/337 /* Forward call to MSGINA. */ 342 338 GWlxDisplayLockedNotice(pWlxContext); 343 339 } … … 347 343 { 348 344 Log(("VBoxGINA::WlxIsLockOk\n")); 349 /* forward call to MSGINA*/345 /* Forward call to MSGINA. */ 350 346 return GWlxIsLockOk(pWlxContext); 351 347 } … … 354 350 { 355 351 Log(("VBoxGINA::WlxWkstaLockedSAS\n")); 356 /* forward call to MSGINA */ 352 353 /* when performing a direct logon without C-A-D, our poller might not be running */ 354 if (!credentialsAvailable()) 355 credentialsPollerCreate(); 356 357 /* Forward call to MSGINA. */ 357 358 return GWlxWkstaLockedSAS(pWlxContext, dwSasType); 358 359 } … … 383 384 Log(("VBoxGINA::WlxLogoff\n")); 384 385 385 /* forward call to MSGINA*/386 /* Forward call to MSGINA. */ 386 387 GWlxLogoff(pWlxContext); 387 388 } … … 392 393 Log(("VBoxGINA::WlxShutdown\n")); 393 394 394 /* forward call to MSGINA*/395 /* Forward call to MSGINA. */ 395 396 GWlxShutdown(pWlxContext, ShutdownType); 396 397 } … … 400 401 * GINA 1.1 entry points 401 402 */ 402 403 403 BOOL WINAPI WlxScreenSaverNotify(PVOID pWlxContext, BOOL *pSecure) 404 404 { 405 405 Log(("VBoxGINA::WlxScreenSaverNotify\n")); 406 406 407 /* forward to MSGINA if present*/407 /* Forward to MSGINA if present. */ 408 408 if (GWlxScreenSaverNotify) 409 409 return GWlxScreenSaverNotify(pWlxContext, pSecure); … … 413 413 } 414 414 415 415 416 BOOL WINAPI WlxStartApplication(PVOID pWlxContext, PWSTR pszDesktopName, 416 417 PVOID pEnvironment, PWSTR pszCmdLine) … … 419 420 pWlxContext, pszDesktopName, pEnvironment, pszCmdLine)); 420 421 421 /* forward to MSGINA if present*/422 /* Forward to MSGINA if present. */ 422 423 if (GWlxStartApplication) 423 424 return GWlxStartApplication(pWlxContext, pszDesktopName, pEnvironment, pszCmdLine); … … 425 426 } 426 427 428 427 429 /* 428 430 * GINA 1.3 entry points … … 432 434 Log(("VBoxGINA::WlxNetworkProviderLoad\n")); 433 435 434 /* forward to MSGINA if present*/436 /* Forward to MSGINA if present. */ 435 437 if (GWlxNetworkProviderLoad) 436 438 return GWlxNetworkProviderLoad(pWlxContext, pNprNotifyInfo); … … 444 446 Log(("VBoxGINA::WlxDisplayStatusMessage\n")); 445 447 446 /* forward to MSGINA if present*/448 /* Forward to MSGINA if present. */ 447 449 if (GWlxDisplayStatusMessage) 448 450 return GWlxDisplayStatusMessage(pWlxContext, hDesktop, dwOptions, pTitle, pMessage); … … 456 458 Log(("VBoxGINA::WlxGetStatusMessage\n")); 457 459 458 /* forward to MSGINA if present*/460 /* Forward to MSGINA if present. */ 459 461 if (GWlxGetStatusMessage) 460 462 return GWlxGetStatusMessage(pWlxContext, pdwOptions, pMessage, dwBufferSize); … … 467 469 Log(("VBoxGINA::WlxRemoveStatusMessage\n")); 468 470 469 /* forward to MSGINA if present*/471 /* Forward to MSGINA if present. */ 470 472 if (GWlxRemoveStatusMessage) 471 473 return GWlxRemoveStatusMessage(pWlxContext); … … 477 479 * GINA 1.4 entry points 478 480 */ 479 480 481 BOOL WINAPI WlxGetConsoleSwitchCredentials(PVOID pWlxContext,PVOID pCredInfo) 481 482 { … … 488 489 } 489 490 491 490 492 VOID WINAPI WlxReconnectNotify(PVOID pWlxContext) 491 493 { 492 494 Log(("VBoxGINA::WlxReconnectNotify\n")); 493 495 494 /* forward to MSGINA if present*/496 /* Forward to MSGINA if present. */ 495 497 if (GWlxReconnectNotify) 496 498 GWlxReconnectNotify(pWlxContext); 497 499 } 498 500 501 499 502 VOID WINAPI WlxDisconnectNotify(PVOID pWlxContext) 500 503 { 501 504 Log(("VBoxGINA::WlxDisconnectNotify\n")); 502 505 503 /* forward to MSGINA if present*/506 /* Forward to MSGINA if present. */ 504 507 if (GWlxDisconnectNotify) 505 508 GWlxDisconnectNotify(pWlxContext); 506 509 } 510
Note:
See TracChangeset
for help on using the changeset viewer.