关于manifest引起的Crash

项目期间,软件在开发机器上运行流畅,但一旦到公共机,来不及启动界面就crash了,于是动用了depends,将缺失的DLL等等都加上去,还是crash,于是乎,火了,难道是引用了一些debug版的DLL?于是乎,把开发环境也装上,还是crash,在木有头绪的时候,注意到depends下面提示到的一些信息,关于开源库DLL出现error的情况,做了一个简单的测试,将出现error的dll拷贝到我机器上,木有报错,于是问题应该是这些DLL调用了不同版本的系统DLL,导致出错了。

 

         沿着这条线路,加上群友点拨,终于追到manifest这个东东。关于manifest的介绍,请看下面的文章 http://blog.sina.com.cn/s/blog_5515f2a20100xb7i.html

 

1.关于manifest:
转贴一个:http://www.zming.net/blog/index_en.php?itemid=183   谢谢作者Ming  
   
  dll是被动态调用的,所以会被若干个程序共享使用的   但是如果dll在应用程序不知道的情况下升级了、或是被另一个程序更改了,就可能会出现问题,即"DLL   Hell"  
   
  随着系统资源越来越丰富,硬盘不那么紧张,所以在XP以后的操作系统中,用新的机制来管理DLL    
  (这种机制,这不仅仅是对于.NET而言,对于普通的Native程序也是一样的)    
   
  Madifest是个XML的描述文件,对于每个DLL有DLL的Manifest文件,对于每个应用程序Application也有自己的Manifest  
   
  对于应用程序而言,Manifest可以是一个和exe文件同一目录下的.manifest文件,也可以是作为一个资源嵌入在exe文件内部的(Embed   Manifest)    
   
  XP以前版本的windows,会像以前那样执行这个exe文件,寻找相应的dll,没有分别  
  Manifest只是个多余的文件或资源,dll文件会直接到system32的目录下查找,并且调用  
   
  而XP以后的操作系统,则会首先读取Manifest,获得exe文件需要调用的DLL列表  
  (此时获得的,并不直接是DLL文件的本身的位置,而是DLL的manifest)    
  操作系统再根据DLL的Manifest去寻找对应的DLL    
  <因此就可能区别不同版本的同一个DLL文件,或是指定一个程序本身Isolated的DLL>  
   
  不过使用Visual   Studio   2005以后的一个新问题是,  
  VS2005带的8.0新版的C运行库(VC   8.0   CRT)文件在XP以后支持manifest的Windows版本中被调用时,  
  将会check一下Application自身的Manifest,否则将会拒绝被调用  
 这也就是说,使用Visual   Studio开发的Application,Manifest将是必不可少的  
  (搞不懂MS为啥要这样设置,所以与VS2003.NET不同了)  
  (后来想想,除了MS自己说的哪些冠冕堂皇的原因,至少这样一来Linux的Wine模拟要麻烦多了)    
   
  除非,你的程序是静态链接的,没有使用dll,只使用了操作系统核心的   Kernel32.dll,   User32.dll,   Ole32.dll,   或ShDocVW.dll   等  
  project的设置必须是Use   Standard   Windows   Libraries、Not   Using   ATL、No   Common   Language   Runtime   support    
  那么你可以不需要考虑Manifest   可以关掉它  
   
  VS2005中Menifest相关的设置:  
      Project/Properties/Linker/Manifest   File/Generate   Manifest       决定是否生成Manifest,如上情况才可以关闭  
      Project/Properties/Linker/Manifest   File/Allow   Isolation                 这个是设置DLL的调用的,并不能决定Manifest是否还是必须的    
      Project/Properties/Manifest   Tool/Input   and   Output/Embed   Manifest     决定Manifest是个单独的文件还是嵌入到exe内的资源

2。问题:问题的产生是因为从VC6的工程到VS2005的移植,正是因为上述原因而可能出现以下链接错误:
1>Linking...
1>CVTRES : fatal error CVT1100: duplicate resource.  type:MANIFEST, name:1, language:0x0409
1>LINK : fatal error LNK1123: failure during conversion to COFF: file invalid or corrupt

解决办法是:打开rc文件,找MANIFEST,所属段或行全去掉

 

 

 

         看完这篇文章后,我就纳闷了,因为加载的DLL是开源项目中的,所以只能给系统打补丁,增加相关的DLL和manifest文件,于是乎,打开万能的UltraEdit,将一个DLL拖进去,翻了翻,终于出现如下几行,如获至宝:

 

  <dependency>

    <dependentAssembly>

      <assemblyIdentity type="win32" name="Microsoft.VC80.CRT" version="8.0.50727.4053" processorArchitecture="x86" publicKeyToken="1fc8b3b9a1e18e3b"></assemblyIdentity>

    </dependentAssembly>

  </dependency>

 

         有dependency这样的关键字,再将Microsoft.VC80.CRT 和 8.0.50727.4053关键字上谷歌百度,很快就有好多的链接,几经尝试,终于解决问题。

 

         问题如下:

         vc7.1~vc9.0都有这样的问题,这是微软搞出来的,名义上说是为了解决dll hell问题,对dll的依赖,不单单是检测文件名\函数或类入口,还检测menifest,而我所引用的DLL是vs2008编译的,所以它遵循这个规则。PS,以后还是要自己找开源代码来编译,这个比较靠谱。

 

         在操作系统,有个winsxs目录,里面的menifest和profix指定dll的相关版本在哪里,然后,你的运行程序,有一个menifest, 你的menifest指定的版本,要在那里面去找,匹配了,并且指定的目录有这个dll,才能加载,而我把dll放在执行目录,没有manifest这个中介是没用的,

 

 

         上面是VC80.CRT中版本8.0.50727.4053,publicKeyToken是1fc8b3b9a1e18e3b的,在下面的网站终于找到了补丁,问题解决,good

 

 

http://support.microsoft.com/kb/973923/zh-cn#appliesto

 

http://technet.microsoft.com/zh-cn/security/bulletin/ms09-035

posted @ 2012-12-08 12:03  Anbit自由之路  阅读(488)  评论(0编辑  收藏  举报