我有以下功能来安装我的Windows c ++服务并使用它多年。最近,我已经努力将其转换为unicode,并进行了一些代码更改。代码仍适用于多字节代码,并创建运行时检查失败#2 - 变量“MyKey”周围的堆栈因unicode而损坏。我认为调用RegSetValueEx可能存在问题,但无法找到原因。欢迎任何建议。在函数调用结束时抛出错误。
static TCHAR* NTSERVICE=_T("MyService");
static TCHAR* svcname=_T("My Service");
void InstallService(char *exename)
{
SC_HANDLE myService, scm;
HKEY MyKey;
TRACE ("Installing service...");
scm = OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE);
if (!scm) {
TRACE ("Failed to open Service Control Manager! (Error code = %d)", GetLastError());
return;
}
DWORD Disposition = 0;
TCHAR modname[256];
GetModuleFileName(NULL, (LPTSTR)modname, sizeof(modname)/sizeof(TCHAR));
myService = CreateService(scm,
(LPCTSTR)NTSERVICE, //Internal service name
(LPCTSTR)svcname, //Show name
SERVICE_ALL_ACCESS, //We want full control
SERVICE_WIN32_OWN_PROCESS, //Let's not mess it up for somebody else..
SERVICE_AUTO_START, //The service requires manual start
SERVICE_ERROR_NORMAL, //Normal handling when error in startup
(LPCTSTR)modname, //Binary file
0, 0, 0, 0, 0); //Misc :)
if (!myService)
{
TRACE (_T("Failed to create the %s service! (Error code =%d)"), NTSERVICE , GetLastError());
CloseServiceHandle(scm);
return;
}
TCHAR keyname[300];
_tcscpy_s(keyname,sizeof(keyname),_T("SYSTEM\\CurrentControlSet\\Services\\"));
_tcscat_s(keyname,300, NTSERVICE);
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE,
(LPCTSTR)keyname,
NULL, //Reserved
NULL, //Class
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
NULL, //Security attributes :)
&MyKey,
&Disposition)
!= ERROR_SUCCESS) {
TRACE(_T("Failed to open registry key!"));
return;
}
TCHAR buffer[1024];
int size = strlen(exename);
MultiByteToWideChar(CP_ACP, 0, (char*)exename, size, buffer, size*2);
buffer[size] = _T('\0');
if (RegSetValueEx(MyKey, _T("Exename"), NULL, REG_SZ, (const BYTE*)buffer,
(size + 1) * sizeof(TCHAR)) != ERROR_SUCCESS)
{
TRACE(_T("Failed to write binary executable name to registry!"));
// printf("Failed to write binary executable name to registry!");
RegCloseKey(MyKey);
return;
}
RegCloseKey(MyKey);
TRACE(_T("Service successfully installed."));
printf("Service successfully installed.");
CloseServiceHandle(myService);
CloseServiceHandle(scm);
}
答案 0 :(得分:1)
GetModuleFileName(NULL, (LPTSTR)modname, sizeof(modname));
_tcscpy_s(keyname,sizeof(keyname),_T("SYSTEM\\CurrentControlSet\\Services\\"));
sizeof
以字节为单位返回大小。由于TCHAR
是wchar_t
且定义了UNICODE,因此字节数是字符数的两倍。
使用
GetModuleFileName(NULL, (LPTSTR)modname, _countof(modname));
_tcscpy_s(keyname,_countof(keyname),_T("SYSTEM\\CurrentControlSet\\Services\\"));
或
GetModuleFileName(NULL, (LPTSTR)modname, sizeof(modname)/sizeof(TCHAR));
_tcscpy_s(keyname,sizeof(keyname)/sizeof(TCHAR),_T("SYSTEM\\CurrentControlSet\\Services\\"));
代替。
答案 1 :(得分:0)
请改为尝试:
static const LPTSTR NTSERVICE = TEXT("MyService");
static const LPTSTR svcname = TEXT("My Service");
void InstallService(char *exename)
{
SC_HANDLE myService, scm;
HKEY MyKey;
TRACE (_T("Installing service..."));
scm = OpenSCManager(0, 0, SC_MANAGER_CREATE_SERVICE);
if (!scm) {
TRACE (_T("Failed to open Service Control Manager! (Error code = %d)"), GetLastError());
return;
}
TCHAR modname[MAX_PATH+1] = {0};
GetModuleFileName(NULL, modname, MAX_PATH);
myService = CreateService(scm,
NTSERVICE, //Internal service name
svcname, //Show name
SERVICE_ALL_ACCESS, //We want full control
SERVICE_WIN32_OWN_PROCESS, //Let's not mess it up for somebody else..
SERVICE_AUTO_START, //The service requires manual start
SERVICE_ERROR_NORMAL, //Normal handling when error in startup
modname, //Binary file
0, 0, 0, 0, 0); //Misc :)
if (!myService)
{
TRACE (_T("Failed to create the %s service! (Error code =%d)"), NTSERVICE, GetLastError());
CloseServiceHandle(scm);
return;
}
TCHAR keyname[300];
_tcscpy_s(keyname, 300, _T("SYSTEM\\CurrentControlSet\\Services\\"));
_tcscat_s(keyname, 300, NTSERVICE);
DWORD Disposition = 0;
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE,
keyname,
NULL, //Reserved
NULL, //Class
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS,
NULL, //Security attributes :)
&MyKey,
&Disposition) != ERROR_SUCCESS)
{
TRACE(_T("Failed to open registry key!"));
CloseServiceHandle(myService);
CloseServiceHandle(scm);
return;
}
TCHAR buffer[MAX_PATH+1] = {0};
#ifdef UNICODE
int len = MultiByteToWideChar(CP_ACP, 0, exename, strlen(exename), buffer, MAX_PATH);
#else
strcpy_s(buffer, MAX_PATH, exename);
int len = strlen(buffer);
#endif
buffer[len] = 0;
if (RegSetValueEx(MyKey,
TEXT("Exename"),
NULL,
REG_SZ,
(const BYTE*)buffer,
(len + 1) * sizeof(TCHAR)) != ERROR_SUCCESS)
{
TRACE(_T("Failed to write binary executable name to registry!"));
RegCloseKey(MyKey);
CloseServiceHandle(myService);
CloseServiceHandle(scm);
return;
}
RegCloseKey(MyKey);
CloseServiceHandle(myService);
CloseServiceHandle(scm);
TRACE(_T("Service successfully installed."));
printf("Service successfully installed.");
}