ImageMagick: win7 | win8 & uac (用户帐户控制) 注册表的一些事

时间:2023-03-09 07:29:48
ImageMagick: win7 | win8 & uac (用户帐户控制) 注册表的一些事

现在用win7,win8的人越来越多了, 程序在一些 win 7, win8 上运行会遇到一些之前没想过的兼容性问题。

比如 64位系统运行32位程序时的注册表重定向,还有因为 uac (用户帐户控制)注册表的重定向等。

ImageMagick 在安装的时候,相关数据写在 HKEY_LOCAL_MACHINE 下面, 这在 xp 系统下,一般不会出什么问题。

但由于在 win7,win8上,由于64位系统或uac的问题,导致注册表的路径重定向, 特别是 uac 问题,要想将数据保存到HKEY_LOCAL_MACHINE 必须要管理员身份运行软件。

经过一天多时间研究,一开始在网上找到一个办法,通过修改 ImageMagick 的环境变量,因为根据ImageMagick的源代码,module.c (537行)

static MagickBooleanType GetMagickModulePath(const char *filename,

MagickModuleType module_type,char *path,ExceptionInfo *exception)

{

char

*module_path;

assert(filename != (const char *) NULL);

(void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",filename);

assert(path != (char *) NULL);

assert(exception != (ExceptionInfo *) NULL);

(void) CopyMagickString(path,filename,MaxTextExtent);

module_path=(char *) NULL;

switch (module_type)

{

case MagickImageCoderModule:

default:

{

(void) LogMagickEvent(ModuleEvent,GetMagickModule(),

"Searching for coder module file \"%s\" ...",filename);

module_path = GetEnvironmentValue("MAGICK_CODER_MODULE_PATH");

#if defined(MAGICKCORE_CODER_PATH)

if (module_path == (char *) NULL)

module_path=AcquireString(MAGICKCORE_CODER_PATH);

#endif

一开始,我在程序开头加了一句:putenv("MAGICK_CODER_MODULE_PATH=c:\\imagemagick\\modules\\coders\\"); //设置环境变量

我以为这样设置后,ImageMagick就不会对注册表有依赖了,可谁知:

printf("MAGICK_CODER_MODULE_PATH:%s\n", getenv("MAGICK_CODER_MODULE_PATH")); //在我的程序中能正常显示的路径

但是 GetEnvironmentValue("MAGICK_CODER_MODULE_PATH"); 返回的是 (null)

我查看了 ImageMagick 的源代码,发现 GetEnvironmentValue() 其实调用的就是 putenv()

可为什么返回的是  null ? 测试发现使用 putenv("MAGICK_CODER_MODULE_PATH=..."); 一点用没有。

再继续研究ImageMagick源代码:

module.c (634行)

#if defined(MAGICKCORE_WINDOWS_SUPPORT)

{

const char

*registery_key;

unsigned char

*key_value;

/*

Locate path via registry key.

*/

switch (module_type)

{

case MagickImageCoderModule:

default:

{

registery_key="CoderModulesPath";

break;

}

case MagickImageFilterModule:

{

registery_key="FilterModulesPath";

break;

}

}

key_value=NTRegistryKeyLookup(registery_key);

if (key_value == (unsigned char *) NULL)

{

ThrowMagickException(exception,GetMagickModule(),ConfigureError,

"RegistryKeyLookupFailed","`%s'",registery_key);

return(MagickFalse);

}

-------------------------------------------------------------------------------

nt-base.c (1813行)

MagickPrivate unsigned char *NTRegistryKeyLookup(const char *subkey)

{

char

package_key[MaxTextExtent];

DWORD

size,

type;

HKEY

registry_key;

LONG

status;

unsigned char

*value;

/*

Look-up base key.

*/

(void) FormatLocaleString(package_key,MaxTextExtent,"SOFTWARE\\%s\\%s\\Q:%d",

MagickPackageName,MagickLibVersionText,MAGICKCORE_QUANTUM_DEPTH);

(void) LogMagickEvent(ConfigureEvent,GetMagickModule(),"%s",package_key);

registry_key=(HKEY) INVALID_HANDLE_VALUE;

status=RegOpenKeyExA(HKEY_LOCAL_MACHINE,package_key,0,KEY_READ,&registry_key);

if (status != ERROR_SUCCESS)

status=RegOpenKeyExA(HKEY_CURRENT_USER,package_key,0,KEY_READ,

&registry_key);

if (status != ERROR_SUCCESS)

{

registry_key=(HKEY) INVALID_HANDLE_VALUE;

return((unsigned char *) NULL);

}

-------------------------------------------------------------------------------------

原来 ImageMagick 搜索注册表的路径有个顺序,先搜索 HKEY_LOCAL_MACHINE, 再搜索 HKEY_CURRENT_USER

看到这里,我想到一个办法,由于 uac 的问题导致普通用户无法写入设置 HKEY_LOCAL_MACHINE,但可以设置 HKEY_CURRENT_USER

只需将相关的数据写入到 HKEY_CURRENT_USER, 这样就不需要管理员身份也能正常的运行了。

经过测试,验证了我的这个想法,问题终于解决了,以后不用烦在 win7,win8上的兼容性问题了。

2014-09-19