Changeset 57961 in vbox for trunk/src/VBox/Additions
- Timestamp:
- Sep 30, 2015 10:11:22 AM (10 years ago)
- svn:sync-xref-src-repo-rev:
- 102949
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Additions/common/VBoxService/VBoxServiceControlSession.cpp
r57960 r57961 1764 1764 /* ppSessionThread is optional. */ 1765 1765 1766 #ifdef DEBUG 1767 PVBOXSERVICECTRLSESSIONTHREAD pSessionCur; 1766 #ifdef VBOX_STRICT 1768 1767 /* Check for existing session in debug mode. Should never happen because of 1769 1768 * Main consistency. */ 1769 PVBOXSERVICECTRLSESSIONTHREAD pSessionCur; 1770 1770 RTListForEach(pList, pSessionCur, VBOXSERVICECTRLSESSIONTHREAD, Node) 1771 1771 { 1772 if (pSessionCur->StartupInfo.uSessionID == pSessionStartupInfo->uSessionID) 1773 { 1774 AssertMsgFailed(("Guest session thread ID=%RU32 (%p) already exists when it should not\n", 1775 pSessionCur->StartupInfo.uSessionID, pSessionCur)); 1776 return VERR_ALREADY_EXISTS; 1777 } 1772 AssertMsgReturn(pSessionCur->StartupInfo.uSessionID != pSessionStartupInfo->uSessionID, 1773 ("Guest session thread ID=%RU32 (%p) already exists when it should not\n", 1774 pSessionCur->StartupInfo.uSessionID, pSessionCur), VERR_ALREADY_EXISTS); 1778 1775 } 1779 1776 #endif … … 1785 1782 s_uCtrlSessionThread = 0; /* Wrap around to not let IPRT freak out. */ 1786 1783 1784 /* 1785 * Allocate and initialize the session thread structure. 1786 */ 1787 1787 PVBOXSERVICECTRLSESSIONTHREAD pSessionThread = 1788 1788 (PVBOXSERVICECTRLSESSIONTHREAD)RTMemAllocZ(sizeof(VBOXSERVICECTRLSESSIONTHREAD)); … … 1790 1790 { 1791 1791 /* Copy over session startup info. */ 1792 memcpy(&pSessionThread->StartupInfo, pSessionStartupInfo, 1793 sizeof(VBOXSERVICECTRLSESSIONSTARTUPINFO)); 1792 memcpy(&pSessionThread->StartupInfo, pSessionStartupInfo, sizeof(VBOXSERVICECTRLSESSIONSTARTUPINFO)); 1794 1793 1795 1794 pSessionThread->fShutdown = false; … … 1805 1804 Assert(!strlen(pSessionThread->StartupInfo.szDomain)); 1806 1805 1807 VBoxServiceVerbose(3, "New anonymous guest session ID=%RU32 created, uFlags=%x, using protocol %RU32\n",1806 VBoxServiceVerbose(3, "New anonymous guest session ID=%RU32 created, fFlags=%x, using protocol %RU32\n", 1808 1807 pSessionStartupInfo->uSessionID, 1809 1808 pSessionStartupInfo->fFlags, … … 1812 1811 else 1813 1812 { 1814 VBoxServiceVerbose(3, "Spawning new guest session ID=%RU32, szUser=%s, szPassword=%s, szDomain=%s, uFlags=%x, using protocol %RU32\n",1813 VBoxServiceVerbose(3, "Spawning new guest session ID=%RU32, szUser=%s, szPassword=%s, szDomain=%s, fFlags=%x, using protocol %RU32\n", 1815 1814 pSessionStartupInfo->uSessionID, 1816 1815 pSessionStartupInfo->szUser, … … 1826 1825 1827 1826 rc = RTCritSectInit(&pSessionThread->CritSect); 1828 AssertRC(rc); 1829 1827 AssertRC(rc); /* We'll check this again further down, right... */ 1828 1829 /** @todo r=bird: split function here, that would simplify failure cleanup... */ 1830 1830 /* 1831 1831 * Spawn a child process for doing the actual session handling. 1832 * Start by assembling the argument list. 1832 1833 */ 1833 1834 char szExeName[RTPATH_MAX]; … … 1862 1863 1863 1864 /* Add same verbose flags as parent process. */ 1864 /** @todo r=bird: how does this stuff work when g_cVerbosity = 0? Also, why do you need rc2 here? Do we actually care about the RC anyway? */ 1865 int rc2 = VINF_SUCCESS; 1866 char szParmVerbose[32] = { 0 }; 1867 for (int i = 0; i < g_cVerbosity && RT_SUCCESS(rc2); i++) 1868 { 1869 if (i == 0) 1870 rc2 = RTStrCat(szParmVerbose, sizeof(szParmVerbose), "-"); 1871 if (RT_FAILURE(rc2)) 1872 break; 1873 rc2 = RTStrCat(szParmVerbose, sizeof(szParmVerbose), "v"); 1874 } 1875 if (RT_SUCCESS(rc2)) 1865 char szParmVerbose[32]; 1866 if (g_cVerbosity > 0) 1867 { 1868 unsigned cVs = RT_MIN(g_cVerbosity, RT_ELEMENTS(szParmVerbose) - 2); 1869 szParmVerbose[0] = '-'; 1870 memset(&szParmVerbose[1], 'v', cVs); 1871 szParmVerbose[1 + cVs] = '\0'; 1876 1872 apszArgs[idxArg++] = szParmVerbose; 1877 1878 /** @todo r=bird: As I already mentioned in the review comment you removed in 1879 * r102561, the log file name handling is rather mangled. You have a 4K stack 1880 * buffer, but you insist on doing 2-5 heap allocations when constructing the 1881 * name. Either you do it in the stack buffer or you do it on the heap. 1882 * 1883 * Now, if you make sure szParmLogFile is, say, 128 byte larger than 1884 * g_szLogFile, you won't need to consider overflows any more. You can easily 1885 * construct the whole thing with a strcpy, RTPathStripSuffix and a sprintf. 1886 */ 1873 } 1887 1874 1888 1875 /* Add log file handling. Each session will have an own 1889 1876 * log file, naming based on the parent log file. */ 1890 #if 11891 char szParmLogFile[RTPATH_MAX];1892 if ( RT_SUCCESS(rc2)1893 && strlen(g_szLogFile))1894 {1895 char *pszLogFile = RTStrDup(g_szLogFile);1896 if (pszLogFile)1897 {1898 char *pszLogSuff = NULL;1899 if (RTPathHasSuffix(pszLogFile))1900 pszLogSuff = RTStrDup(RTPathSuffix(pszLogFile));1901 RTPathStripSuffix(pszLogFile);1902 char *pszLogNewSuffix;1903 #ifndef DEBUG1904 if (RTStrAPrintf(&pszLogNewSuffix, "-%RU32-%s",1905 pSessionStartupInfo->uSessionID,1906 pSessionStartupInfo->szUser) < 0)1907 {1908 rc2 = VERR_NO_MEMORY;1909 }1910 #else /* DEBUG */1911 /* Include the session thread ID in the log file name. */1912 if (RTStrAPrintf(&pszLogNewSuffix, "-%RU32-%RU32-%s",1913 pSessionStartupInfo->uSessionID,1914 s_uCtrlSessionThread,1915 pSessionStartupInfo->szUser) < 0)1916 {1917 rc2 = VERR_NO_MEMORY;1918 }1919 #endif /* DEBUG */1920 else1921 {1922 rc2 = RTStrAAppend(&pszLogFile, pszLogNewSuffix);1923 if (RT_SUCCESS(rc2) && pszLogSuff)1924 rc2 = RTStrAAppend(&pszLogFile, pszLogSuff);1925 if (RT_SUCCESS(rc2))1926 RTStrPrintf(szParmLogFile, sizeof(szParmLogFile), "--logfile=%s", pszLogFile);1927 1928 RTStrFree(pszLogNewSuffix);1929 }1930 if (RT_FAILURE(rc2))1931 VBoxServiceError("Error building session logfile string for session %RU32 (user %s), rc=%Rrc\n",1932 pSessionStartupInfo->uSessionID, pSessionStartupInfo->szUser, rc2);1933 if (pszLogSuff)1934 RTStrFree(pszLogSuff);1935 RTStrFree(pszLogFile);1936 }1937 if (RT_SUCCESS(rc2))1938 apszArgs[idxArg++] = szParmLogFile;1939 1940 rc = rc2;1941 }1942 else if (RT_FAILURE(rc2))1943 rc = rc2;1944 #else1945 1877 char szParmLogFile[sizeof(g_szLogFile) + 128]; 1946 1878 if (g_szLogFile) … … 1961 1893 apszArgs[idxArg++] = szParmLogFile; 1962 1894 } 1963 #endif /* alternative version */ 1895 1964 1896 #ifdef DEBUG 1965 1897 VBoxServiceVerbose(4, "Argv building rc=%Rrc, session flags=%x\n", rc, g_Session.fFlags); … … 1978 1910 { 1979 1911 VBoxServiceVerbose(4, "Spawning parameters:\n"); 1980 1981 idxArg = 0; 1982 while (apszArgs[idxArg]) 1983 VBoxServiceVerbose(4, "\t%s\n", apszArgs[idxArg++]); 1984 } 1985 1986 uint32_t uProcFlags = RTPROC_FLAGS_SERVICE 1987 #ifdef RT_OS_WINDOWS 1988 /* Make sure to also load the profile data on a Windows guest. */ 1989 | RTPROC_FLAGS_PROFILE /** @todo Not implemented for non-Windows yet. */ 1990 #endif 1991 | RTPROC_FLAGS_HIDDEN; /** @todo More flags from startup info? */ 1912 for (idxArg = 0; apszArgs[idxArg]; idxArg++) 1913 VBoxServiceVerbose(4, "\t%s\n", apszArgs[idxArg]); 1914 } 1915 1992 1916 /* 1993 * C reate the session process' environment block.1917 * Configure standard handles and finally create the process. 1994 1918 */ 1995 if (RT_SUCCESS(rc)) 1996 { 1997 if (g_cVerbosity > 3) 1998 { 1999 /** @todo At the moment a session process does not have the ability to use the 2000 * per-session environment variables itself, only the session's guest 2001 * processes do so. Implement that later, also needs tweaking of 2002 * VbglR3GuestCtrlSessionGetOpen(). */ 2003 RTENV hEnv = NIL_RTENV; 2004 rc = RTEnvClone(&hEnv, RTENV_DEFAULT); 2005 if (RT_SUCCESS(rc)) 2006 { 2007 VBoxServiceVerbose(4, "Environment variables:\n"); 2008 2009 uint32_t cVars = RTEnvCountEx(hEnv); 2010 for (uint32_t iVar = 0; iVar < cVars; iVar++) 2011 { 2012 char szVar[_1K]; 2013 char szValue[_16K]; 2014 rc2 = RTEnvGetByIndexEx(hEnv, iVar, szVar, sizeof(szVar), szValue, sizeof(szValue)); 2015 if (RT_SUCCESS(rc2)) 2016 VBoxServiceVerbose(4, "\t%s=%s\n", szVar, szValue); 2017 else if (rc2 == VERR_BUFFER_OVERFLOW) 2018 VBoxServiceVerbose(4, "\t%s=%s [VERR_BUFFER_OVERFLOW]\n", szVar, szValue); 2019 else 2020 { 2021 VBoxServiceVerbose(4, "\tUnable to enumerate environment variable #%RU32: %Rrc\n", iVar, rc2); 2022 /* Keep going. */ 2023 } 2024 } 2025 2026 RTEnvDestroy(hEnv); 2027 } 2028 } 2029 } 1919 uint32_t fProcCreate = RTPROC_FLAGS_SERVICE 1920 #ifdef RT_OS_WINDOWS /** @todo do on unix too! */ 1921 /* Make sure to also load the profile data on a Windows guest. */ 1922 | RTPROC_FLAGS_PROFILE 1923 #endif 1924 | RTPROC_FLAGS_HIDDEN; /** @todo More flags from startup info? */ 2030 1925 2031 1926 #if 0 /* Pipe handling not needed (yet). */ … … 2056 1951 2057 1952 if (RT_SUCCESS(rc)) 2058 rc = RTProcCreateEx(pszExeName, apszArgs, hEnv, uProcFlags,1953 rc = RTProcCreateEx(pszExeName, apszArgs, hEnv, fProcCreate, 2059 1954 pSession->StdIn.phChild, pSession->StdOut.phChild, pSession->StdErr.phChild, 2060 1955 !fAnonymous ? pSession->StartupInfo.szUser : NULL, … … 2090 1985 hStdOutAndErr.enmType = RTHANDLETYPE_FILE; 2091 1986 2092 rc = RTProcCreateEx(pszExeName, apszArgs, RTENV_DEFAULT, uProcFlags,1987 rc = RTProcCreateEx(pszExeName, apszArgs, RTENV_DEFAULT, fProcCreate, 2093 1988 &hStdIn, &hStdOutAndErr, &hStdOutAndErr, 2094 1989 !fAnonymous ? pSessionThread->StartupInfo.szUser : NULL, … … 2108 2003 if (RT_SUCCESS(rc)) 2109 2004 { 2110 /* Start session thread. */ 2005 /* 2006 * Start session the thread. 2007 */ 2111 2008 rc = RTThreadCreateF(&pSessionThread->Thread, gstcntlSessionThread, 2112 2009 pSessionThread /*pvUser*/, 0 /*cbStack*/, … … 2115 2012 { 2116 2013 VBoxServiceError("Creating session thread failed, rc=%Rrc\n", rc); 2014 /** @todo r=bird: what about the process we created? Perhaps a 2015 * RTProcTerminate would be in order? */ 2117 2016 } 2118 2017 else … … 2145 2044 if (RT_FAILURE(rc)) 2146 2045 { 2046 /** @todo r=bird: what about deleting the critsect? You're leaking event 2047 * semaphore handles here. */ 2147 2048 RTMemFree(pSessionThread); 2148 2049 }
Note:
See TracChangeset
for help on using the changeset viewer.