home *** CD-ROM | disk | FTP | other *** search
/ DOS/V Power Report 2001 June / VPR0106A.BIN / OLS / TAR32212 / tar32212.lzh / tar32_2 / src / tar32.cpp < prev    next >
C/C++ Source or Header  |  2000-12-25  |  17KB  |  550 lines

  1. /*
  2.     tar32.cpp
  3.         tar/tgz/tbz/bz/gz archive manipulation class.
  4.         by Yoshioka Tsuneo(QWF00133@nifty.ne.jp)
  5. */
  6. /*    
  7.     このファイルの利用条件:
  8.         このソースファイルの利用制限は一切ありません。
  9.         ソースの一部、全部を商用、非商用など目的に
  10.         かかわりなく他のプログラムで自由に使用できます。
  11.         パブリック・ドメイン・ソフトウェアと同様に扱えます。
  12.     
  13.     プログラマ向けの要望(制限ではありません):
  14.         ソース中に改善すべき点があればお知らせください。
  15.         ソースコード中にバグを見つけた場合は報告してください。
  16.         直した部分などありましたら教えてください。
  17.         断片的な情報でも結構です。
  18.         このファイルを利用した場合はなるべく教えてください。
  19. */
  20. /*
  21.     LICENSE of this file:
  22.         There is no restriction for using this file.
  23.         You can use this file in your software for any purpose.
  24.         In other words, you can use this file as Public Domain Software.
  25.  
  26.     RECOMMENDATION for Programmer(not restriction):
  27.         If you find points to improve code, please report me.
  28.         If you find bug in source code, please report me.
  29.         If you fixed bug, please teach me.
  30.         I want any trivial information.
  31.         If you use this file, please report me.
  32. */
  33. #include "tar.h"
  34. #include "cpio.h"
  35. #include "ar.h"
  36.  
  37. #include "tar32.h"
  38. #include "util.h"
  39. #include "tar32dll.h" // CTar32Exception
  40. #include "tar32api.h" // API ERROR codes
  41. #include <stdio.h>
  42. #include <sys/types.h>
  43. #include <sys/stat.h> 
  44.  
  45. #include <algorithm>
  46. #include <string>
  47. #include <memory>
  48. #include <fstream>
  49. using namespace std;
  50.  
  51. CTar32::CTar32()
  52. {
  53.     m_pfile = NULL;
  54.     m_archive_type = 0;
  55.     m_filecount = 0;
  56.     m_write_mode = false;
  57.     m_error_code = 0;
  58.     longfilenames_buf = NULL;
  59. }
  60. CTar32::~CTar32()
  61. {
  62.     close();
  63. }
  64. int CTar32::s_get_archive_type(const char *arcfile)
  65. {
  66.     auto_ptr<ITarArcFile> pfile(ITarArcFile::s_open(arcfile,"rb",ARCHIVETYPE_AUTO));
  67.     if(pfile.get()==NULL){return -1;}
  68.  
  69.     //int archive_type = ITarArcFile::s_get_archive_type();
  70.     //if(archive_type == -1){return -1;}
  71.     
  72.     // HEADER tar_header;
  73.     int ret;
  74.  
  75.     int archive_type = pfile->get_archive_type();
  76.  
  77.     union archive_header{
  78.         HEADER tar;
  79.         new_cpio_header cpio;
  80.         ar_first_hdr ar;
  81.     };
  82.     archive_header arc_header;
  83.     ret = pfile->read(&arc_header,sizeof(arc_header));
  84.     if(ret >= sizeof(arc_header.tar)
  85.         && arc_header.tar.compsum() == strtol(arc_header.tar.dbuf.chksum, NULL, 8)){
  86.         switch(archive_type){
  87.         case ARCHIVETYPE_NORMAL:
  88.             archive_type = ARCHIVETYPE_TAR;break;
  89.         case ARCHIVETYPE_GZ:
  90.             archive_type = ARCHIVETYPE_TARGZ;break;
  91.         case ARCHIVETYPE_Z:
  92.             archive_type = ARCHIVETYPE_TARZ;break;
  93.         case ARCHIVETYPE_BZ2:
  94.             archive_type = ARCHIVETYPE_TARBZ2;break;
  95.         }
  96.     }else if(ret >= sizeof(arc_header.cpio)
  97.         && arc_header.cpio.magic_check()){
  98.         switch(archive_type){
  99.         case ARCHIVETYPE_NORMAL:
  100.             archive_type = ARCHIVETYPE_CPIO;break;
  101.         case ARCHIVETYPE_GZ:
  102.             archive_type = ARCHIVETYPE_CPIOGZ;break;
  103.         case ARCHIVETYPE_Z:
  104.             archive_type = ARCHIVETYPE_CPIOZ;break;
  105.         case ARCHIVETYPE_BZ2:
  106.             archive_type = ARCHIVETYPE_CPIOBZ2;break;
  107.         }
  108.     }else if(ret >= sizeof(arc_header.ar)
  109.         && (memcmp(arc_header.ar.magic,"!<arch>\012",8) == 0  || memcmp(arc_header.ar.magic,"!<bout>\012",8) == 0)
  110.         && arc_header.ar.hdr.magic_check()){
  111.         switch(archive_type){
  112.         case ARCHIVETYPE_NORMAL:
  113.             archive_type = ARCHIVETYPE_AR;break;
  114.         case ARCHIVETYPE_GZ:
  115.             archive_type = ARCHIVETYPE_ARGZ;break;
  116.         case ARCHIVETYPE_Z:
  117.             archive_type = ARCHIVETYPE_ARZ;break;
  118.         case ARCHIVETYPE_BZ2:
  119.             archive_type = ARCHIVETYPE_ARBZ2;break;
  120.         }
  121.     }
  122.     return archive_type;
  123. }
  124. bool CTar32::open(const char *arcfile,const char *mode,int archive_type /*= ARCHIVETYPE_AUTO*/)
  125. {
  126.     m_archive_type = archive_type;
  127.     m_write_mode = (strchr(mode,'w') != NULL);
  128.     if((!strchr(mode,'w')) && archive_type == ARCHIVETYPE_AUTO){ // if read mode
  129.         m_archive_type = s_get_archive_type(arcfile);
  130.         if(m_archive_type == -1 || m_archive_type == ARCHIVETYPE_NORMAL){
  131.             return false;
  132.         }
  133.     }
  134.     m_pfile = ITarArcFile::s_open(arcfile,mode,m_archive_type);
  135.     if(!m_pfile){return false;}
  136.     return true;
  137. }
  138. bool CTar32::close()
  139. {
  140.     if(m_write_mode && m_pfile && m_error_code==0){
  141.         if(m_archive_type == ARCHIVETYPE_TAR 
  142.             || m_archive_type == ARCHIVETYPE_TARGZ 
  143.             || m_archive_type == ARCHIVETYPE_TARZ 
  144.             || m_archive_type == ARCHIVETYPE_TARBZ2){
  145.             // If success to create tar file, append 1024(512*2) byte null block to the end of file.
  146.             char buf[1024/*512 * 2*/]; memset(buf,0,sizeof(buf));
  147.             m_pfile->write(buf,sizeof(buf));
  148.         }
  149.     }
  150.     if(m_pfile){
  151.         m_pfile->close();
  152.         delete m_pfile;
  153.         m_pfile = NULL;
  154.     }
  155.     if(longfilenames_buf){
  156.         delete [] longfilenames_buf;
  157.         longfilenames_buf = NULL;
  158.     }
  159.     return true;
  160. }
  161. bool CTar32::readdir(CTar32FileStatus *pstat)
  162. {
  163.     HEADER tar_header;
  164.     int ret;
  165.     CTar32FileStatus stat;
  166.  
  167.     
  168.     if(m_archive_type == ARCHIVETYPE_TAR 
  169.         || m_archive_type == ARCHIVETYPE_TARGZ 
  170.         || m_archive_type == ARCHIVETYPE_TARZ 
  171.         || m_archive_type == ARCHIVETYPE_TARBZ2){
  172.         {
  173.             ret = m_pfile->read(&tar_header,sizeof(tar_header));
  174.             if(ret == 0){return false;}
  175.             if(ret != sizeof(tar_header)){
  176.                 throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);
  177.             }
  178.             HEADER zero_header;
  179.             memset(&zero_header,0,sizeof(zero_header));
  180.             if(memcmp(&tar_header,&zero_header,sizeof(tar_header)) == 0){
  181.                 return false;
  182.             }
  183.             if((unsigned long)tar_header.compsum() != (unsigned long)strtol(tar_header.dbuf.chksum , NULL, 8)){
  184.                 throw CTar32Exception("tar header checksum error.",ERROR_HEADER_CRC);
  185.             }
  186.         }
  187.         stat.filename    =    tar_header.dbuf.name;
  188.         stat.original_size        =    strtol(tar_header.dbuf.size, NULL, 8);
  189.         stat.blocksize  =   512;
  190.         if(tar_header.dbuf.typeflag == LONGLINK){    // tar_header.dbuf.name == "././@LongLink"
  191.             char longfilename[2000] = "";
  192.             int readsize = ((stat.original_size-1)/512+1)*512;
  193.             ret = m_pfile->read(longfilename, readsize);
  194.             if(ret == 0){
  195.                 throw CTar32Exception("can't get filename(LongLink)",ERROR_HEADER_BROKEN);
  196.             }
  197.             longfilename[stat.original_size]='\0';
  198.             {
  199.                 ret = m_pfile->read(&tar_header,sizeof(tar_header));
  200.                 if(ret == 0){return false;}
  201.                 if(ret != sizeof(tar_header)){
  202.                     throw CTar32Exception("can't read tar header(LongLink)",ERROR_HEADER_BROKEN);
  203.                 }
  204.                 HEADER zero_header;
  205.                 memset(&zero_header,0,sizeof(zero_header));
  206.                 if(memcmp(&tar_header,&zero_header,sizeof(tar_header)) == 0){
  207.                     return false;
  208.                 }
  209.                 if((unsigned long)tar_header.compsum() != (unsigned long)strtol(tar_header.dbuf.chksum , NULL, 8)){
  210.                     throw CTar32Exception("tar header checksum error.(LongLink)",ERROR_HEADER_CRC);
  211.                 }
  212.             }
  213.             stat.filename = longfilename;
  214.             stat.original_size        =    strtol(tar_header.dbuf.size, NULL, 8);
  215.         }
  216.         
  217.         stat.mode        =   strtol(tar_header.dbuf.mode, NULL, 8);
  218.         stat.uid        =   strtol(tar_header.dbuf.uid , NULL, 8);
  219.         stat.gid        =   strtol(tar_header.dbuf.gid , NULL, 8);
  220.         stat.mtime        =   strtol(tar_header.dbuf.mtime , NULL, 8);
  221.         stat.chksum        =   strtol(tar_header.dbuf.chksum , NULL, 8);
  222.         stat.typeflag    =   tar_header.dbuf.typeflag;
  223.         stat.linkname    =    tar_header.dbuf.linkname;
  224.         strncpy(stat.magic_version, tar_header.dbuf.magic,8);
  225.         strncpy(stat.uname, tar_header.dbuf.uname, 32);
  226.         strncpy(stat.gname, tar_header.dbuf.gname, 32);
  227.         stat.devmajor    =   strtol(tar_header.dbuf.devmajor , NULL, 8);
  228.         stat.devminor    =   strtol(tar_header.dbuf.devminor , NULL, 8);
  229.         stat.atime        =   strtol(tar_header.dbuf.atime , NULL, 8);
  230.         stat.ctime        =   strtol(tar_header.dbuf.ctime , NULL, 8);
  231.         stat.offset        =   strtol(tar_header.dbuf.offset , NULL, 8);
  232.     }else if(m_archive_type == ARCHIVETYPE_CPIO
  233.         || m_archive_type == ARCHIVETYPE_CPIOGZ
  234.         || m_archive_type == ARCHIVETYPE_CPIOZ
  235.         || m_archive_type == ARCHIVETYPE_CPIOBZ2){
  236.         new_cpio_header cpio_header;
  237.         {
  238.             ret = m_pfile->read(&cpio_header,sizeof(cpio_header));
  239.             if(ret == 0){return false;}
  240.             if(ret != sizeof(cpio_header)){
  241.                 throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);
  242.             }
  243.             new_cpio_header zero_header;
  244.             memset(&zero_header,0,sizeof(zero_header));
  245.             if(! cpio_header.magic_check()){
  246.                 throw CTar32Exception("tar header checksum error.",ERROR_HEADER_CRC);
  247.             }
  248.         }
  249.         int fnamelen;
  250.         int dum;
  251.         int nret = sscanf(((char*)&cpio_header)+6, "%8lx%8lx%8lx%8lx%8lx%8lx%8lx%8lx%8lx%8lx%8lx%8lx%8lx"
  252.             ,&dum, &stat.mode, &stat.uid, &stat.gid, &dum, &stat.mtime
  253.             ,&stat.original_size,&stat.devmajor, &stat.devminor, &dum, &dum
  254.             ,&fnamelen, &stat.chksum);
  255.         if(nret != 13){
  256.                 throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);
  257.         }
  258.  
  259.         if(fnamelen<=0 || fnamelen>=1000){return false;}
  260.         char fname[1000];
  261.         int fnamelen2 = (((fnamelen+sizeof(new_cpio_header) -1)/4)+1)*4 - sizeof(new_cpio_header);    /* 4 byte padding for ("new cpio header" + "filename") */
  262.         m_pfile->read(fname,fnamelen2);
  263.         stat.filename    =    fname;
  264.         stat.blocksize        = 4;
  265.         if(stat.filename == "TRAILER!!!"){
  266.             return false;
  267.         }
  268.         if((stat.mode & _S_IFMT) == _S_IFDIR){
  269.             stat.filename = stat.filename + "/";
  270.         }
  271.     }else if(m_archive_type == ARCHIVETYPE_AR
  272.         || m_archive_type == ARCHIVETYPE_ARGZ
  273.         || m_archive_type == ARCHIVETYPE_ARZ
  274.         || m_archive_type == ARCHIVETYPE_ARBZ2){
  275.         if(m_filecount == 0){
  276.             char magic[8];
  277.             ret = m_pfile->read(magic,8);    /* skip "!<arch>\012" */
  278.         }
  279.         ar_hdr header;
  280.         {
  281.             ret = m_pfile->read(&header,sizeof(header));
  282.             if(ret == 0){return false;}
  283.             if(ret != sizeof(header)){
  284.                 throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);
  285.             }
  286.             ar_hdr zero_header;
  287.             memset(&zero_header,0,sizeof(zero_header));
  288.             if(! header.magic_check()){
  289.                 throw CTar32Exception("tar header checksum error.",ERROR_HEADER_CRC);
  290.             }
  291.         }
  292.         char *p = strchr(header.name,' '); *p = '\0';            // cut from ' '
  293.         if(strcmp(header.name,"//")!=0 && p-1 > header.name && strrchr(header.name,'/') == p-1){ *(p-1) = '\0';}    // cut last '/'
  294.         stat.filename = header.name;
  295.         stat.mtime = strtol(header.date,NULL,10);
  296.         stat.uid = strtol(header.uid,NULL,10);
  297.         stat.gid = strtol(header.gid,NULL,10);
  298.         stat.mode = strtol(header.mode,NULL,8);
  299.         stat.original_size = strtol(header.size,NULL,10);
  300.         stat.blocksize        = 2;
  301.  
  302.         if(stat.filename == "/"){
  303.             char buf[1000];
  304.             sprintf(buf,"root%d",m_filecount);
  305.             stat.filename = buf;
  306.         }
  307.         if(stat.filename == "//"){
  308.             /* long file name support */
  309.             CTar32InternalFile file;
  310.             m_currentfile_status = stat;
  311.             if(!file.open(this)){throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);}
  312.             longfilenames_buf = new char[stat.original_size]; // (char*)malloc(stat.original_size);
  313.             int n = file.read(longfilenames_buf,stat.original_size);
  314.             if(n!=stat.original_size){throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);}
  315.             file.close();
  316.             CTar32FileStatus nextstat;
  317.             bool bRet = readdir(&nextstat); m_filecount --;
  318.             stat = nextstat;
  319.             //if(!bRet || stat.filename != "/0"){
  320.             //    throw CTar32Exception("can't read tar header",ERROR_HEADER_BROKEN);
  321.             //}
  322.             //stat.filename = longfilename;
  323.         }
  324.         {
  325.             /* if use longfilename, substitute "/\d+" to filename */
  326.             int bytes; int num;
  327.             int n = sscanf(stat.filename.c_str(), "/%d%n", &bytes, &num);
  328.             int len = stat.filename.length();
  329.             if(n == 1 && num == len && longfilenames_buf){
  330.                 stat.filename = longfilenames_buf+bytes;
  331.             }
  332.         }
  333.  
  334.     }else{
  335.         if(m_filecount != 0){return false;}
  336.         stat.filename        = m_pfile->get_orig_filename();
  337.         stat.original_size    = m_pfile->get_orig_size();
  338.         stat.mtime            = m_pfile->get_mtime();
  339.         stat.blocksize        = 1;
  340.     }
  341.     m_filecount ++;
  342.     *pstat = stat;
  343.     m_currentfile_status = stat;
  344.     return true;
  345. }
  346. bool CTar32::readskip()
  347. {
  348.     CTar32InternalFile file;
  349.     if(! file.open(this)){
  350.         return false;
  351.     }
  352.     if(! file.close()){
  353.         return false;
  354.     }
  355. #if 0
  356.     int size;
  357.     int ret;
  358.  
  359.     if(m_currentfile_status.size == 0){return true;}
  360.  
  361.     size = (((m_currentfile_status.size-1)/512)+1)*512;
  362.     ret = m_pfile->seek(size, SEEK_CUR);
  363.     if(ret == -1){return false;}
  364. #endif
  365.     return true;
  366. }
  367. bool CTar32::extract(const char *fname_extract_to)
  368. {
  369.     string fname;
  370.     if(fname_extract_to){
  371.         fname = fname_extract_to;
  372.     }else{
  373.         fname = m_currentfile_status.filename;
  374.     }
  375.     int filesize = m_currentfile_status.original_size;
  376.     const int buf_size = 4096;
  377.     char buf[buf_size];
  378.  
  379.     CTar32InternalFile file;
  380.     file.open(this);
  381.  
  382.     //string dirname = get_dirname(fname.c_str());
  383.     //mkdir_recursive(dirname.c_str());
  384.     mkdir_recursive(get_dirname(fname.c_str()).c_str());
  385.  
  386.     ofstream fs_w;
  387.     fs_w.open(fname.c_str(), ios::out|ios::binary);
  388.     if(fs_w.fail()){return false;}
  389.     //FILE *fp_w = fopen(fname.c_str(), "wb");
  390.     //if(fp_w == NULL){
  391.     //    return false;
  392.     //}
  393.     int readsize = 0;
  394.     while(readsize<filesize){
  395.         int nextreadsize = min((int)filesize-readsize,(int)sizeof(buf));
  396.         int n = file.read(buf,nextreadsize);
  397.         fs_w.write(buf,nextreadsize);
  398.         if(fs_w.fail()){return false;}
  399.         //fwrite(buf,1,ret,fp_w);
  400.         readsize += n;
  401.         if(n != nextreadsize){/*fclose(fp_w);*/return false;}
  402.     }
  403.     //fclose(fp_w);
  404.     return true;
  405. }
  406.  
  407. bool CTar32::addheader(const CTar32FileStatus &stat)
  408. {
  409.     int ret;
  410.     
  411.     if(m_archive_type == ARCHIVETYPE_TAR 
  412.         || m_archive_type == ARCHIVETYPE_TARGZ 
  413.         || m_archive_type == ARCHIVETYPE_TARZ 
  414.         || m_archive_type == ARCHIVETYPE_TARBZ2){
  415.  
  416.         HEADER tar_header;
  417.         {
  418.             HEADER *pblock = &tar_header;
  419.             memset(pblock, 0, sizeof(*pblock));
  420.             const CTar32FileStatus *p = &stat;
  421.             string fname = p->filename;
  422.  
  423.             if(fname.length() > sizeof(pblock->dbuf.name)/*100*/){
  424.                 CTar32FileStatus tmpstat = stat;
  425.                 tmpstat.filename = "././@LongLink";
  426.                 tmpstat.typeflag = LONGLINK;
  427.                 tmpstat.original_size = fname.length();
  428.                 bool bret = addheader(tmpstat);
  429.                 char filename[2000];
  430.                 strcpy(filename, fname.c_str());
  431.                 int writesize = ((fname.length() - 1)/512+1)*512;
  432.                 ret = m_pfile->write(filename, writesize);
  433.                 if(ret != writesize){
  434.                     throw CTar32Exception("LongLink filename write error", ERROR_CANNOT_WRITE);
  435.                 }
  436.             }
  437.             strncpy(pblock->dbuf.name, fname.c_str(),NAMSIZ-1); /* tarnt 0.96->0.97 */
  438.             sprintf(pblock->dbuf.mode, "%6o ", (unsigned int)p->mode);
  439.             sprintf(pblock->dbuf.uid, "%06o ",p->uid);
  440.             sprintf(pblock->dbuf.gid, "%06o ",p->gid);
  441.             sprintf(pblock->dbuf.size, "%11lo ", p->original_size);
  442.             sprintf(pblock->dbuf.mtime, "%11lo ", p->mtime);
  443.             pblock->dbuf.typeflag = p->typeflag;
  444.             memcpy(pblock->dbuf.magic, p->magic_version, sizeof(p->magic_version));
  445.             strncpy(pblock->dbuf.uname, p->uname, sizeof pblock->dbuf.uname);
  446.             sprintf(pblock->dbuf.chksum, "%6o ", pblock->compsum());
  447.         }
  448.  
  449.         ret = m_pfile->write(&tar_header,sizeof(tar_header));
  450.         if(ret != sizeof(tar_header)){
  451.             throw CTar32Exception("header write error", ERROR_CANNOT_WRITE);
  452.         }
  453.     }else{
  454.         ;
  455.     }
  456.     m_currentfile_status = stat;
  457.     return true;
  458. }
  459. bool CTar32::addbody(const char *file)
  460. {
  461.     struct _stat st;
  462.     if(_stat(file, &st) == -1){
  463.         char msg[1000];
  464.         sprintf(msg, "can't read file[%s]", file);
  465.         throw CTar32Exception(msg, ERROR_CANNOT_READ);
  466.     }
  467.     if(st.st_size == 0){return true;}
  468.     //FILE *fp_r;
  469.     //fp_r = fopen(file, "rb");
  470.     ifstream fs_r;
  471.     fs_r.open(file,ios::in|ios::binary);
  472.     //if(!fp_r){
  473.     if(fs_r.fail()){
  474.         throw CTar32Exception("can't read file", ERROR_CANNOT_READ);
  475.     }
  476.     int size = 0;
  477.  
  478.     int n;
  479.     char buf[4096];
  480.     //while((n = fread(buf,1,sizeof(buf),fp_r))>0){
  481.     while(fs_r.read(buf,sizeof(buf)),(n=fs_r.gcount())>0){
  482.         int m = m_pfile->write(buf, n);
  483.         if(m>0){size += m;}
  484.         if(n!=m){
  485.             throw CTar32Exception("can't write to arcfile", ERROR_CANNOT_WRITE);
  486.         }
  487.     }
  488.  
  489.     if(m_archive_type == ARCHIVETYPE_TAR 
  490.         || m_archive_type == ARCHIVETYPE_TARGZ 
  491.         || m_archive_type == ARCHIVETYPE_TARZ 
  492.         || m_archive_type == ARCHIVETYPE_TARBZ2){
  493.         /* padding 512-byte block */
  494.         int writesize;
  495.         if(size%512 == 0){
  496.             writesize = 0;
  497.         }else{
  498.             writesize = 512 - size%512;
  499.         }
  500.         memset(buf,0,writesize);
  501.         m_pfile->write(buf, writesize);
  502.     }
  503.     return true;
  504. }
  505.  
  506.  
  507.  
  508.  
  509.  
  510.  
  511.  
  512. //
  513. //  CTar32InternalFile
  514. //
  515. //
  516. CTar32InternalFile::CTar32InternalFile(){
  517.     m_pfile = NULL;
  518. }
  519. CTar32InternalFile::~CTar32InternalFile(){
  520.     if(m_pfile)close();
  521. }
  522. // open after CTar32::readdir() or CTar32()::addheader()
  523. bool CTar32InternalFile::open(CTar32 *pTar32){
  524.     m_pfile = pTar32->m_pfile;
  525.     m_size = pTar32->m_currentfile_status.original_size;
  526.     m_readsize = 0;
  527.     m_blocksize = pTar32->m_currentfile_status.blocksize;
  528.     return true;
  529. }
  530. int  CTar32InternalFile::write(void *buf, int size){
  531.     return m_pfile->write(buf, size);
  532. }
  533. int  CTar32InternalFile::read(void *buf, int size){
  534.     int n = m_pfile->read(buf, m_size==-1 ? size : min(size, m_size-m_readsize));
  535.     m_readsize += n;
  536.     return n;
  537. }
  538. bool CTar32InternalFile::close(){
  539.     int size = m_size;
  540.     bool bret = true;
  541.     if(size != -1 && size!=0){
  542.         size = (((size-1)/m_blocksize)+1) * m_blocksize;
  543.         size = size - m_readsize;
  544.         int ret = m_pfile->seek(size, SEEK_CUR);
  545.         bret = (ret != -1);
  546.     }
  547.     m_pfile = NULL;
  548.     return bret;
  549. }
  550.