cmCPackTGZGenerator.cxx 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. /*=========================================================================
  2. Program: CMake - Cross-Platform Makefile Generator
  3. Module: $RCSfile$
  4. Language: C++
  5. Date: $Date$
  6. Version: $Revision$
  7. Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
  8. See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
  9. This software is distributed WITHOUT ANY WARRANTY; without even
  10. the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
  11. PURPOSE. See the above copyright notices for more information.
  12. =========================================================================*/
  13. #include "cmCPackTGZGenerator.h"
  14. #include "cmake.h"
  15. #include "cmGlobalGenerator.h"
  16. #include "cmLocalGenerator.h"
  17. #include "cmSystemTools.h"
  18. #include "cmMakefile.h"
  19. #include "cmGeneratedFileStream.h"
  20. #include "cmCPackLog.h"
  21. #include <cmsys/SystemTools.hxx>
  22. #include <cmzlib/zutil.h>
  23. #include <libtar/libtar.h>
  24. #include <memory> // auto_ptr
  25. #include <fcntl.h>
  26. #include <errno.h>
  27. //----------------------------------------------------------------------
  28. class cmCPackTGZGeneratorForward
  29. {
  30. public:
  31. static int GenerateHeader(cmCPackTGZGenerator* gg, std::ostream* os)
  32. {
  33. return gg->GenerateHeader(os);
  34. }
  35. };
  36. //----------------------------------------------------------------------
  37. cmCPackTGZGenerator::cmCPackTGZGenerator()
  38. {
  39. }
  40. //----------------------------------------------------------------------
  41. cmCPackTGZGenerator::~cmCPackTGZGenerator()
  42. {
  43. }
  44. static const size_t cmCPackTGZ_Data_BlockSize = 16384;
  45. //----------------------------------------------------------------------
  46. class cmCPackTGZ_Data
  47. {
  48. public:
  49. cmCPackTGZ_Data(cmCPackTGZGenerator* gen) :
  50. OutputStream(0), Generator(gen),
  51. CompressionLevel(Z_DEFAULT_COMPRESSION) {}
  52. std::ostream* OutputStream;
  53. cmCPackTGZGenerator* Generator;
  54. char CompressedBuffer[cmCPackTGZ_Data_BlockSize];
  55. int CompressionLevel;
  56. z_stream ZLibStream;
  57. uLong CRC;
  58. };
  59. //----------------------------------------------------------------------
  60. extern "C" {
  61. int cmCPackTGZ_Data_Open(void *client_data, const char* name, int oflags,
  62. mode_t mode);
  63. ssize_t cmCPackTGZ_Data_Write(void *client_data, void *buff, size_t n);
  64. int cmCPackTGZ_Data_Close(void *client_data);
  65. }
  66. //----------------------------------------------------------------------
  67. int cmCPackTGZ_Data_Open(void *client_data, const char* pathname,
  68. int, mode_t)
  69. {
  70. cmCPackTGZ_Data *mydata = (cmCPackTGZ_Data*)client_data;
  71. mydata->ZLibStream.zalloc = Z_NULL;
  72. mydata->ZLibStream.zfree = Z_NULL;
  73. mydata->ZLibStream.opaque = Z_NULL;
  74. int strategy = Z_DEFAULT_STRATEGY;
  75. if ( deflateInit2(&mydata->ZLibStream, mydata->CompressionLevel,
  76. Z_DEFLATED, -MAX_WBITS, DEF_MEM_LEVEL, strategy) != Z_OK )
  77. {
  78. return -1;
  79. }
  80. cmGeneratedFileStream* gf = new cmGeneratedFileStream;
  81. // Open binary
  82. gf->Open(pathname, false, true);
  83. mydata->OutputStream = gf;
  84. if ( !*mydata->OutputStream )
  85. {
  86. return -1;
  87. }
  88. if ( !cmCPackTGZGeneratorForward::GenerateHeader(mydata->Generator,gf))
  89. {
  90. return -1;
  91. }
  92. mydata->CRC = crc32(0L, Z_NULL, 0);
  93. return 0;
  94. }
  95. //----------------------------------------------------------------------
  96. ssize_t cmCPackTGZ_Data_Write(void *client_data, void *buff, size_t n)
  97. {
  98. cmCPackTGZ_Data *mydata = (cmCPackTGZ_Data*)client_data;
  99. mydata->ZLibStream.avail_in = n;
  100. mydata->ZLibStream.next_in = reinterpret_cast<Bytef*>(buff);
  101. do {
  102. mydata->ZLibStream.avail_out = cmCPackTGZ_Data_BlockSize;
  103. mydata->ZLibStream.next_out
  104. = reinterpret_cast<Bytef*>(mydata->CompressedBuffer);
  105. // no bad return value
  106. int ret = deflate(&mydata->ZLibStream, (n?Z_NO_FLUSH:Z_FINISH));
  107. if(ret == Z_STREAM_ERROR)
  108. {
  109. return 0;
  110. }
  111. size_t compressedSize
  112. = cmCPackTGZ_Data_BlockSize - mydata->ZLibStream.avail_out;
  113. mydata->OutputStream->write(
  114. reinterpret_cast<const char*>(mydata->CompressedBuffer),
  115. compressedSize);
  116. } while ( mydata->ZLibStream.avail_out == 0 );
  117. if ( !*mydata->OutputStream )
  118. {
  119. return 0;
  120. }
  121. if ( n )
  122. {
  123. mydata->CRC = crc32(mydata->CRC, reinterpret_cast<Bytef *>(buff), n);
  124. }
  125. return n;
  126. }
  127. //----------------------------------------------------------------------
  128. int cmCPackTGZ_Data_Close(void *client_data)
  129. {
  130. cmCPackTGZ_Data *mydata = (cmCPackTGZ_Data*)client_data;
  131. cmCPackTGZ_Data_Write(client_data, 0, 0);
  132. char buffer[8];
  133. int n;
  134. uLong x = mydata->CRC;
  135. for (n = 0; n < 4; n++) {
  136. buffer[n] = (int)(x & 0xff);
  137. x >>= 8;
  138. }
  139. x = mydata->ZLibStream.total_in;
  140. for (n = 0; n < 4; n++) {
  141. buffer[n+4] = (int)(x & 0xff);
  142. x >>= 8;
  143. }
  144. mydata->OutputStream->write(buffer, 8);
  145. (void)deflateEnd(&mydata->ZLibStream);
  146. delete mydata->OutputStream;
  147. mydata->OutputStream = 0;
  148. return (0);
  149. }
  150. //----------------------------------------------------------------------
  151. int cmCPackTGZGenerator::InitializeInternal()
  152. {
  153. this->SetOptionIfNotSet("CPACK_INCLUDE_TOPLEVEL_DIRECTORY", "1");
  154. return this->Superclass::InitializeInternal();
  155. }
  156. //----------------------------------------------------------------------
  157. int cmCPackTGZGenerator::CompressFiles(const char* outFileName,
  158. const char* toplevel, const std::vector<std::string>& files)
  159. {
  160. cmCPackLogger(cmCPackLog::LOG_DEBUG, "Toplevel: " << toplevel << std::endl);
  161. cmCPackTGZ_Data mydata(this);
  162. TAR *t;
  163. char buf[TAR_MAXPATHLEN];
  164. char pathname[TAR_MAXPATHLEN];
  165. tartype_t gztype = {
  166. (openfunc_t)cmCPackTGZ_Data_Open,
  167. (closefunc_t)cmCPackTGZ_Data_Close,
  168. (readfunc_t)0,
  169. (writefunc_t)cmCPackTGZ_Data_Write,
  170. &mydata
  171. };
  172. // Ok, this libtar is not const safe. for now use auto_ptr hack
  173. char* realName = new char[ strlen(outFileName) + 1 ];
  174. std::auto_ptr<char> realNamePtr(realName);
  175. strcpy(realName, outFileName);
  176. int flags = O_WRONLY | O_CREAT;
  177. if (tar_open(&t, realName,
  178. &gztype,
  179. flags, 0644,
  180. (this->GeneratorVerbose?TAR_VERBOSE:0)
  181. | 0) == -1)
  182. {
  183. cmCPackLogger(cmCPackLog::LOG_ERROR, "Problem with tar_open(): "
  184. << strerror(errno) << std::endl);
  185. return 0;
  186. }
  187. std::vector<std::string>::const_iterator fileIt;
  188. for ( fileIt = files.begin(); fileIt != files.end(); ++ fileIt )
  189. {
  190. std::string rp = cmSystemTools::RelativePath(toplevel, fileIt->c_str());
  191. strncpy(pathname, fileIt->c_str(), sizeof(pathname));
  192. pathname[sizeof(pathname)-1] = 0;
  193. strncpy(buf, rp.c_str(), sizeof(buf));
  194. buf[sizeof(buf)-1] = 0;
  195. if (tar_append_tree(t, pathname, buf) != 0)
  196. {
  197. cmCPackLogger(cmCPackLog::LOG_ERROR,
  198. "Problem with tar_append_tree(\"" << buf << "\", \""
  199. << pathname << "\"): "
  200. << strerror(errno) << std::endl);
  201. tar_close(t);
  202. return 0;
  203. }
  204. }
  205. if (tar_append_eof(t) != 0)
  206. {
  207. cmCPackLogger(cmCPackLog::LOG_ERROR, "Problem with tar_append_eof(): "
  208. << strerror(errno) << std::endl);
  209. tar_close(t);
  210. return 0;
  211. }
  212. if (tar_close(t) != 0)
  213. {
  214. cmCPackLogger(cmCPackLog::LOG_ERROR, "Problem with tar_close(): "
  215. << strerror(errno) << std::endl);
  216. return 0;
  217. }
  218. return 1;
  219. }
  220. //----------------------------------------------------------------------
  221. int cmCPackTGZGenerator::GenerateHeader(std::ostream* os)
  222. {
  223. const int gz_magic[2] = {0x1f, 0x8b}; /* gzip magic header */
  224. char header[11];
  225. sprintf(header, "%c%c%c%c%c%c%c%c%c%c", gz_magic[0], gz_magic[1],
  226. Z_DEFLATED, 0 /*flags*/, 0,0,0,0 /*time*/, 0 /*xflags*/, OS_CODE);
  227. os->write(header, 10);
  228. return 1;
  229. }