| 
			
	
	
		                            原创: 星战紫辉  http://www.cppblog.com/rawdata 2009-2-23              关键字: 打印 SPL EMF 文件格式                 问题:              Windows的假脱机打印会在Windows\System32\spool\PRINERS目录下生成.spl和.shd文件,其中的打印内 容存贮在.spl文件中,但是.spl文件格式似乎未公开,那么如何才能将未知的.spl文件剥离成.emf文件呢?               首先,让我们了解一下Windows打印机制:               这是微软的官网的一副打印流程图片:                               其中ISV是应用软件接口,IHV是硬件接口,左边是XP的打印模型,右边是Vista最新的XPS打印模 型,但两者可以互相转换,具有良好的兼容性。不过,这里暂时只关心XP系统的打印过程。              网络打印过程图:                                           但是这些图似乎还不够详细,那么请看下面一副:(摘录于论文:《基于关键字匹配的打印数据截获 系统》):                                基本的思路是: 打印过程发生时,GDI模块和打印驱动(由打印机厂商提供)进行基本的数据交换,在假 脱机设置环境下,生成打印机命令文件:.spl或.emf文件,作为一个打印池的作业,然后Windows后台打印线 程处理打印作业,将数据文件送至打印机打印,打印完删除该打印文件。                 好,现在回到正题:.spl文件该如何剥离成.emf呢?看一个例子:                 在WinHex中打开一个.spl文件:                  参考:  http://www.undocprint.org/formats/winspool/spl  中一些打印结构的定义。               首先,.spl文件都是以0x00010000签名开头,然后一个DWORD 是emf相关区的文件偏移,第3个 DWORD是文档描述字符串(UNICODE)的文件偏移,第4个DWORD 描述的是端口说明字符串(UNICODE)。大 致结构如下:                             文件尾就是这个样子:                            当定位到0x50的文件位置,读取2个DWORD数据之后,就是.emf文件开始了。.emf文件格式是公开的,而 且非常简单,是一系列EMR_XXX开口结构的紧密排列,通常以EMR_HEADER(0x01)开头,以EMR_EOF (0x0E)结尾。其实我们根本没有必要去解析.emf文件格式,Windows  SDK有专门显示.emf文件的API,3个函数就 搞定:                       HENHMETAFILE hEMF = GetEnhMetaFile("EMF_DumpOK.emf");                       PlayEnhMetaFile (dc.m_hDC, hEMF, &rc) ;                          DeleteEnhMetaFile (hEMF) ; 然后.spl文件还有一些东西,我现在还没有解析出来,但是.emf文件已经剥离出来了,后面的可以先不理它。                   然后,开始写程序喽,因为比较简单,所以代码有点随便哦~~:)                                  http://www.cppblog.com/rawdata/  星绽紫辉                   程序截图如下:                          1 2
  3
  #include <windows.h> 4
  #include <winspool.h> 5
  #include <stdio.h> 6
  #include   <locale.h> 7
  #include <tchar.h> 8
  #include <iostream> 9
  using namespace std; 10
  11
  12
  BOOL AnalyseFile(const char* pszFileName); 13
  14
  void PFT(const char* pszInfo,DWORD dwData) 15
    { 16
  printf("%s: 0x%08X\n",pszInfo,dwData); 17
  } 18
  19
  void PFM(const char* pszInfo) 20
    { 21
  printf("%s\n",pszInfo); 22
  } 23
  24
  void UPFM(const wchar_t pszInfo[]) 25
    { 26
  wprintf(L"%s\n",pszInfo); 27
  } 28
  29
  static char* ID_Func[] = 30
    { 31
  "EMR_HEADER", 32
  "EMR_POLYBEZIER", 33
  "EMR_POLYGON", 34
  "EMR_POLYLINE", 35
  "EMR_POLYBEZIERTO", 36
  "EMR_POLYLINETO", 37
  "EMR_POLYPOLYLINE", 38
  "EMR_POLYPOLYGON", 39
  "EMR_SETWINDOWEXTEX", 40
  "EMR_SETWINDOWORGEX", 41
  "EMR_SETVIEWPORTEXTEX", 42
  "EMR_SETVIEWPORTORGEX", 43
  "EMR_SETBRUSHORGEX", 44
  "EMR_EOF", 45
  "EMR_SETPIXELV", 46
  "EMR_SETMAPPERFLAGS", 47
  "EMR_SETMAPMODE", 48
  "EMR_SETBKMODE", 49
  "EMR_SETPOLYFILLMODE", 50
  "EMR_SETROP2", 51
  "EMR_SETSTRETCHBLTMODE", 52
  "EMR_SETTEXTALIGN", 53
  "EMR_SETCOLORADJUSTMENT", 54
  "EMR_SETTEXTCOLOR", 55
  "EMR_SETBKCOLOR", 56
  "EMR_OFFSETCLIPRGN", 57
  "EMR_MOVETOEX", 58
  "EMR_SETMETARGN", 59
  "EMR_EXCLUDECLIPRECT", 60
  "EMR_INTERSECTCLIPRECT", 61
  "EMR_SCALEVIEWPORTEXTEX", 62
  "EMR_SCALEWINDOWEXTEX", 63
  "EMR_SAVEDC", 64
  "EMR_RESTOREDC", 65
  "EMR_SETWORLDTRANSFORM", 66
  "EMR_MODIFYWORLDTRANSFORM", 67
  "EMR_SELECTOBJECT", 68
  "EMR_CREATEPEN", 69
  "EMR_CREATEBRUSHINDIRECT", 70
  "EMR_DELETEOBJECT", 71
  "EMR_ANGLEARC", 72
  "EMR_ELLIPSE", 73
  "EMR_RECTANGLE", 74
  "EMR_ROUNDRECT", 75
  "EMR_ARC", 76
  "EMR_CHORD", 77
  "EMR_PIE", 78
  "EMR_SELECTPALETTE", 79
  "EMR_CREATEPALETTE", 80
  "EMR_SETPALETTEENTRIES", 81
  "EMR_RESIZEPALETTE", 82
  "EMR_REALIZEPALETTE", 83
  "EMR_EXTFLOODFILL", 84
  "EMR_LINETO", 85
  "EMR_ARCTO", 86
  "EMR_POLYDRAW", 87
  "EMR_SETARCDIRECTION", 88
  "EMR_SETMITERLIMIT", 89
  "EMR_BEGINPATH", 90
  "EMR_ENDPATH", 91
  "EMR_CLOSEFIGURE", 92
  "EMR_FILLPATH", 93
  "EMR_STROKEANDFILLPATH", 94
  "EMR_STROKEPATH", 95
  "EMR_FLATTENPATH", 96
  "EMR_WIDENPATH", 97
  "EMR_SELECTCLIPPATH", 98
  "EMR_ABORTPATH", 99
  "69--Unknown", 100
  101
  "EMR_GDICOMMENT", 102
  "EMR_FILLRGN", 103
  "EMR_FRAMERGN", 104
  "EMR_INVERTRGN", 105
  "EMR_PAINTRGN ", 106
  "EMR_EXTSELECTCLIPRGN", 107
  "EMR_BITBLT ", 108
  "EMR_STRETCHBLT", 109
  "EMR_MASKBLT", 110
  "EMR_PLGBLT", 111
  "EMR_SETDIBITSTODEVICE", 112
  "EMR_STRETCHDIBITS", 113
  "EMR_EXTCREATEFONTINDIRECTW", 114
  "EMR_EXTTEXTOUTA ", 115
  "EMR_EXTTEXTOUTW", 116
  "EMR_POLYBEZIER16", 117
  "EMR_POLYGON16 ", 118
  "EMR_POLYLINE16 ", 119
  "EMR_POLYBEZIERTO16", 120
  "EMR_POLYLINETO16 ", 121
  "EMR_POLYPOLYLINE16", 122
  "EMR_POLYPOLYGON16", 123
  "EMR_POLYDRAW16 ", 124
  "EMR_CREATEMONOBRUSH ", 125
  "EMR_CREATEDIBPATTERNBRUSHPT", 126
  "EMR_EXTCREATEPEN", 127
  "EMR_POLYTEXTOUTA ", 128
  "EMR_POLYTEXTOUTW", 129
  "EMR_SETICMMODE  ", 130
  "EMR_CREATECOLORSPACE", 131
  "EMR_SETCOLORSPACE ", 132
  "EMR_DELETECOLORSPACE ", 133
  "EMR_GLSRECORD ", 134
  "EMR_GLSBOUNDEDRECORD", 135
  "EMR_PIXELFORMAT", 136
  "EMR_RESERVED_105 ", 137
  "EMR_RESERVED_106 ", 138
  "EMR_RESERVED_107", 139
  "EMR_RESERVED_108 ", 140
  "EMR_RESERVED_109", 141
  "EMR_RESERVED_110 ", 142
  "EMR_COLORCORRECTPALETTE", 143
  "EMR_SETICMPROFILEA ", 144
  "EMR_SETICMPROFILEW ", 145
  "EMR_ALPHABLEND", 146
  "EMR_SETLAYOUT ", 147
  "EMR_TRANSPARENTBLT", 148
  "EMR_RESERVED_117 ", 149
  "EMR_GRADIENTFILL", 150
  "EMR_RESERVED_119 ", 151
  "EMR_RESERVED_120", 152
  "EMR_COLORMATCHTOTARGETW", 153
  "EMR_CREATECOLORSPACEW" 154
  }; 155
  156
  int main() 157
    { 158
  setlocale(LC_ALL,""); 159
  160
  const char* pszFileName = "C:\\Documents and Settings\\joe\\桌面\\1\\00053.SPL"; 161
  162
  if(!AnalyseFile(pszFileName)) 163
  PFM("Analyse File Failed!"); 164
  else 165
  PFM("Analyse File Successed Completed!"); 166
  167
  return 0; 168
  } 169
  170
  171
  BOOL AnalyseFile(const char* pszFileName) 172
    { 173
  BOOL bRet = FALSE; 174
  175
  DWORD dwStartPos = 0; 176
  177
  FILE* pFile = fopen(pszFileName,"rb"); 178
  179
  if(!pFile) 180
     { 181
  PFM("Open File Failed!"); 182
  return bRet; 183
  } 184
  185
  PFM("Begin Analyse  "); 186
  187
  PFM("[1] Begin to read SPL HeaderInfo:"); 188
  189
   /**//* =======================Headers================================ */ 190
  DWORD dwTmp = 0; 191
  192
  fseek(pFile,0,0); 193
  194
  fread(&dwTmp,sizeof(DWORD),1,pFile); 195
  196
  PFT("签名",dwTmp); 197
  198
  199
  fread(&dwTmp,sizeof(DWORD),1,pFile); 200
  201
  dwStartPos = dwTmp; 202
  203
  PFT("正文信息偏移:",dwTmp); 204
  205
  fread(&dwTmp,sizeof(DWORD),1,pFile); 206
  207
  PFT("文档信息偏移(UNICODE):",dwTmp); 208
  209
  long pos = ftell(pFile); 210
  211
  fseek(pFile,dwTmp,SEEK_SET); 212
  213
   wchar_t pszInfo[256] =  {0}; 214
  pszInfo[0] = L'('; 215
   216
  WORD wTmp; 217
  for(int i = 1;;i++) 218
     { 219
  fread(&wTmp,sizeof(wTmp),1,pFile); 220
  221
  if(!wTmp) 222
  break; 223
  224
  memcpy((char*)&pszInfo[i],&wTmp,sizeof(wTmp)); 225
  } 226
  pszInfo[i] = L')'; 227
  UPFM(pszInfo); 228
  229
  fseek(pFile,pos,SEEK_SET); 230
  231
  fread(&dwTmp,sizeof(DWORD),1,pFile); 232
  233
  PFT("打印端口信息偏移(UNICODE):",dwTmp); 234
  235
  fseek(pFile,dwTmp,SEEK_SET); 236
  237
  memset(pszInfo,0,sizeof(wchar_t)*256); 238
  pszInfo[0] = L'('; 239
  for(i = 1;;i++) 240
     { 241
  fread(&wTmp,sizeof(wTmp),1,pFile); 242
  243
  if(!wTmp) 244
  break; 245
  246
  memcpy((char*)&pszInfo[i],&wTmp,sizeof(wTmp)); 247
  } 248
  pszInfo[i] = L')'; 249
  UPFM(pszInfo); 250
  251
   /**//* ======================== Unknown datas ================================= */ 252
  PFM("[2] Begin to read SPL Unknown Datas:"); 253
  254
  fseek(pFile,dwStartPos,SEEK_SET); 255
  256
  fread(&dwTmp,sizeof(DWORD),1,pFile); 257
  258
  PFT("未知数据",dwTmp); 259
  260
  fread(&dwTmp,sizeof(DWORD),1,pFile); 261
  262
  PFT("未知数据",dwTmp); 263
  264
   /**//* ======================== Record datas ================================= */ 265
  PFM("[3] Begin to read Record Datas:"); 266
  267
  DWORD dwTmp2 = 0; 268
  for(int i=0;;i++) 269
     { 270
  pos = ftell(pFile); 271
  272
  fread(&dwTmp,sizeof(DWORD),1,pFile); 273
  274
  fread(&dwTmp2,sizeof(DWORD),1,pFile); 275
  276
   277
  printf("index: (%04d)  type: 0x%04X  size: %04d  0x%08X (%s)\n",i,dwTmp,dwTmp2,pos,ID_Func[dwTmp-1]); 278
  279
  //        printf("位置: %08X",pos); 280
  281
  //        printf("(%s)\n",ID_Func[dwTmp]); 282
  283
  if(dwTmp == 0x0E) 284
     { 285
  //            printf("index: (%04d)  type: 0x%04X  size: %04d  0x%08X  (End)\n",i,dwTmp,dwTmp2,pos,); 286
  PFM("End of Record Datas."); 287
  break; 288
  } 289
  290
  fseek(pFile,pos+dwTmp2,SEEK_SET); 291
  } 292
  293
  if(pFile) fclose(pFile); 294
  bRet = TRUE; 295
  296
  return bRet; 297
  } 298
  299
  300
  301
  302
  303
  304
  305
  306
  307
  308
  309
                有了以上的分析,你应该很容易写一个spl To  EMF 文件格式的程序了。               如果代码有什么谬误或者Bug,请留言或者EmailToMe:  xiaolu69soft@yahoo.com.cn              祝你好运~               rawdata                      
	   |