/* * * This program 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 program 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 program; if not, write to the Free Software Foundation, * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * In addition, as a special exception, the author gives permission to * link the code of this program with the Half-Life Game Engine ("HL * Engine") and Modified Game Libraries ("MODs") developed by Valve, * L.L.C ("Valve"). You must obey the GNU General Public License in all * respects for all of the code used other than the HL Engine and MODs * from Valve. If you modify this file, you may extend this exception * to your version of the file, but you are not obligated to do so. If * you do not wish to do so, delete this exception statement from your * version. * */ #include "precompiled.h" class CCommandLine: public ICommandLine { public: CCommandLine(); virtual ~CCommandLine(); void CreateCmdLine(const char *commandline); void CreateCmdLine(int argc, const char *argv[]); const char *GetCmdLine() const; // Check whether a particular parameter exists const char *CheckParm(const char *psz, char **ppszValue = nullptr) const; void RemoveParm(const char *pszParm); void AppendParm(const char *pszParm, const char *pszValues); void SetParm(const char *pszParm, const char *pszValues); void SetParm(const char *pszParm, int iValue); // When the commandline contains @name, it reads the parameters from that file void LoadParametersFromFile(const char *&pSrc, char *&pDst, int maxDestLen); private: // Copy of actual command line char *m_pszCmdLine; }; CCommandLine g_CmdLine; ICommandLine *cmdline = &g_CmdLine; ICommandLine *CommandLine() { return &g_CmdLine; } CCommandLine::CCommandLine() { m_pszCmdLine = nullptr; } CCommandLine::~CCommandLine() { if (m_pszCmdLine) { delete [] m_pszCmdLine; m_pszCmdLine = nullptr; } } char *CopyString(const char *src) { if (!src) return nullptr; char *out = (char *)new char[Q_strlen(src) + 1]; Q_strcpy(out, src); return out; } // Creates a command line from the arguments passed in void CCommandLine::CreateCmdLine(int argc, const char *argv[]) { char cmdline[4096] = ""; const int MAX_CHARS = sizeof(cmdline) - 1; for (int i = 0; i < argc; ++i) { if (Q_strchr(argv[i], ' ')) { Q_strlcat(cmdline, "\""); Q_strlcat(cmdline, argv[i]); Q_strlcat(cmdline, "\""); } else { Q_strlcat(cmdline, argv[i]); } Q_strlcat(cmdline, " "); } cmdline[Q_strlen(cmdline)] = '\0'; CreateCmdLine(cmdline); } void CCommandLine::LoadParametersFromFile(const char *&pSrc, char *&pDst, int maxDestLen) { // Suck out the file name char szFileName[ MAX_PATH ]; char *pOut; char *pDestStart = pDst; // Skip the @ sign pSrc++; pOut = szFileName; while (*pSrc && *pSrc != ' ') { *pOut++ = *pSrc++; #if 0 if ((pOut - szFileName) >= (MAX_PATH - 1)) break; #endif } *pOut = '\0'; // Skip the space after the file name if (*pSrc) pSrc++; // Now read in parameters from file FILE *fp = fopen(szFileName, "r"); if (fp) { char c; c = (char)fgetc(fp); while (c != EOF) { // Turn return characters into spaces if (c == '\n') c = ' '; *pDst++ = c; #if 0 // Don't go past the end, and allow for our terminating space character AND a terminating null character. if ((pDst - pDestStart) >= (maxDestLen - 2)) break; #endif // Get the next character, if there are more c = (char)fgetc(fp); } // Add a terminating space character *pDst++ = ' '; fclose(fp); } else { printf("Parameter file '%s' not found, skipping...", szFileName); } } // Purpose: Create a command line from the passed in string // Note that if you pass in a @filename, then the routine will read settings from a file instead of the command line void CCommandLine::CreateCmdLine(const char *commandline) { if (m_pszCmdLine) { delete[] m_pszCmdLine; m_pszCmdLine = nullptr; } char szFull[4096]; char *pDst = szFull; const char *pSrc = commandline; bool allowAtSign = true; while (*pSrc) { if (*pSrc == '@') { if (allowAtSign) { LoadParametersFromFile(pSrc, pDst, sizeof(szFull) - (pDst - szFull)); continue; } } allowAtSign = isspace(*pSrc) != 0; #if 0 // Don't go past the end. if ((pDst - szFull) >= (sizeof(szFull) - 1)) break; #endif *pDst++ = *pSrc++; } *pDst = '\0'; int len = Q_strlen(szFull) + 1; m_pszCmdLine = new char[len]; Q_memcpy(m_pszCmdLine, szFull, len); } // Purpose: Remove specified string ( and any args attached to it ) from command line void CCommandLine::RemoveParm(const char *pszParm) { if (!m_pszCmdLine) return; if (!pszParm || *pszParm == '\0') return; // Search for first occurrence of pszParm char *p, *found; char *pnextparam; int n; int curlen; p = m_pszCmdLine; while (*p) { curlen = Q_strlen(p); found = Q_strstr(p, pszParm); if (!found) break; pnextparam = found + 1; while (pnextparam && *pnextparam && (*pnextparam != '-') && (*pnextparam != '+')) pnextparam++; if (pnextparam && *pnextparam) { // We are either at the end of the string, or at the next param. Just chop out the current param. // # of characters after this param. n = curlen - (pnextparam - p); Q_memcpy(found, pnextparam, n); found[n] = '\0'; } else { // Clear out rest of string. n = pnextparam - found; Q_memset(found, 0, n); } } // Strip and trailing ' ' characters left over. while (1) { int curpos = Q_strlen(m_pszCmdLine); if (curpos == 0 || m_pszCmdLine[ curpos - 1 ] != ' ') break; m_pszCmdLine[curpos - 1] = '\0'; } } // Purpose: Append parameter and argument values to command line void CCommandLine::AppendParm(const char *pszParm, const char *pszValues) { int nNewLength = 0; char *pCmdString; // Parameter. nNewLength = Q_strlen(pszParm); // Values + leading space character. if (pszValues) nNewLength += Q_strlen(pszValues) + 1; // Terminal 0; nNewLength++; if (!m_pszCmdLine) { m_pszCmdLine = new char[ nNewLength ]; Q_strcpy(m_pszCmdLine, pszParm); if (pszValues) { Q_strcat(m_pszCmdLine, " "); Q_strcat(m_pszCmdLine, pszValues); } return; } // Remove any remnants from the current Cmd Line. RemoveParm(pszParm); nNewLength += Q_strlen(m_pszCmdLine) + 1 + 1; pCmdString = new char[ nNewLength ]; Q_memset(pCmdString, 0, nNewLength); Q_strcpy(pCmdString, m_pszCmdLine); // Copy old command line. Q_strcat(pCmdString, " "); // Put in a space Q_strcat(pCmdString, pszParm); if (pszValues) { Q_strcat(pCmdString, " "); Q_strcat(pCmdString, pszValues); } // Kill off the old one delete[] m_pszCmdLine; // Point at the new command line. m_pszCmdLine = pCmdString; } void CCommandLine::SetParm(const char *pszParm, const char *pszValues) { RemoveParm(pszParm); AppendParm(pszParm, pszValues); } void CCommandLine::SetParm(const char *pszParm, int iValue) { char buf[64]; Q_snprintf(buf, sizeof(buf), "%d", iValue); SetParm(pszParm, buf); } // Purpose: Search for the parameter in the current commandline const char *CCommandLine::CheckParm(const char *psz, char **ppszValue) const { static char sz[128] = ""; if (!m_pszCmdLine) return nullptr; if (ppszValue) *ppszValue = nullptr; char *pret = Q_strstr(m_pszCmdLine, psz); if (!pret || !ppszValue) return pret; // find the next whitespace char *p1 = pret; do { ++p1; } while (*p1 != ' ' && *p1); int i = 0; char *p2 = p1 + 1; do { if (p2[i] == '\0' || p2[i] == ' ') break; sz[i] = p2[i]; i++; } while (i < sizeof(sz)); sz[i] = '\0'; *ppszValue = sz; return pret; } const char *CCommandLine::GetCmdLine() const { return m_pszCmdLine; }