home *** CD-ROM | disk | FTP | other *** search
/ DOS/V Power Report 2003 November / VPR0311.ISO / OLS / TAR32223 / tar32223.lzh / tar32_2 / src / util.cpp < prev    next >
C/C++ Source or Header  |  2003-01-17  |  7KB  |  243 lines

  1. /*
  2.     Utility functions, classes for Tar32.dll
  3.         by Yoshioka Tsuneo(QWF00133@nifty.ne.jp)
  4. */
  5. /*    
  6.     このファイルの利用条件:
  7.         このソースファイルの利用制限は一切ありません。
  8.         ソースの一部、全部を商用、非商用など目的に
  9.         かかわりなく他のプログラムで自由に使用できます。
  10.         パブリック・ドメイン・ソフトウェアと同様に扱えます。
  11.     
  12.     プログラマ向けの要望(制限ではありません):
  13.         ソース中に改善すべき点があればお知らせください。
  14.         ソースコード中にバグを見つけた場合は報告してください。
  15.         直した部分などありましたら教えてください。
  16.         断片的な情報でも結構です。
  17.         このファイルを利用した場合はなるべく教えてください。
  18. */
  19. /*
  20.     LICENSE of this file:
  21.         There is no restriction for using this file.
  22.         You can use this file in your software for any purpose.
  23.         In other words, you can use this file as Public Domain Software.
  24.  
  25.     RECOMMENDATION for Programmer(not restriction):
  26.         If you find points to improve code, please report me.
  27.         If you find bug in source code, please report me.
  28.         If you fixed bug, please teach me.
  29.         I want any trivial information.
  30.         If you use this file, please report me.
  31. */
  32. #include "util.h"
  33. #include <string>
  34. #include <direct.h> // _mkdir
  35. #include <wtypes.h>
  36. #include <winnls.h>
  37. #include <io.h>
  38. #include <mbstring.h>
  39.  
  40. int mkdir_recursive(const char *dirname_)
  41. {
  42.     string dirname = dirname_;
  43.     int pos = 0;
  44.     int ret;
  45.     
  46.     while(1){
  47.         string make_dirname;
  48.         char *ptr = (char*)_mbspbrk((const unsigned char*)dirname.c_str()+pos,(const unsigned char*)"/\\");
  49.  
  50.         if(ptr==NULL){
  51.             break;
  52.         }
  53.  
  54.         pos = ptr - dirname.c_str();
  55.         make_dirname = dirname.substr(0,pos);
  56.         ret = _mkdir(make_dirname.c_str());
  57.         pos++;
  58.     }
  59.     ret = _mkdir(dirname.c_str());
  60.     return 0;
  61. }
  62. string get_dirname(const char *pathname)
  63. {
  64.     string filename = pathname;
  65.     //int pos1 = filename.rfind('/');
  66.     //int pos2 = filename.rfind('\\');
  67.     //int pos = max(pos1 ,pos2);
  68.     //if(pos == string::npos){return "";}
  69.     //string dirname = filename.substr(0, pos);
  70.     // fixed by tsuneo 2001.05.14
  71.     char *p = (char*)max(_mbsrchr((const unsigned char*)pathname,'/'),_mbsrchr((const unsigned char*)pathname,'\\'));
  72.     if(! p){
  73.         return "";
  74.     }
  75.     string dirname = filename.substr(0, p - pathname);
  76.     return dirname;
  77. }
  78. string get_filename(const char *pathname)
  79. {
  80.  
  81.     char *p = (char*)max(_mbsrchr((const unsigned char*)pathname,'/'),_mbsrchr((const unsigned char*)pathname,'\\'));
  82.     if(p){
  83.         return string(p+1);
  84.     }else{
  85.         return string(pathname);
  86.     }
  87. }
  88. bool is_absolute_pathname(const char *fname)
  89. {
  90.     if(*fname == '\\'){return true;}
  91.     if(isalpha(*fname) && fname[1] == ':' && fname[2] == '\\'){return true;}
  92.     return false;
  93. }
  94. string make_pathname(const char *dirname_, const char *filename_)
  95. {
  96.     if(*dirname_ == '\0'){return string(filename_);}
  97.     if(is_absolute_pathname(filename_)){
  98.         return string(filename_);
  99.     }
  100.     string pathname = dirname_;
  101.     // if(_mbsrchr((const unsigned char *)dirname_,'\\') == (const unsigned char *)dirname_ + _mbslen((const unsigned char *)dirname_) -1){
  102.     if(_mbsrchr((const unsigned char *)dirname_,'\\') == (const unsigned char *)dirname_ + strlen(dirname_) -1){    // fixed by tsuneo 2001.05.15
  103.         ;
  104.     }else{
  105.         pathname.append(1, '\\');
  106.     }
  107.     pathname.append(filename_);
  108.     return pathname;
  109. }
  110.  
  111.  
  112. class dbcschar{
  113. public:
  114.     dbcschar(const char *str){
  115.         if(str == NULL){highbyte = 0; lowbyte = 0;return;}
  116.         if(::IsDBCSLeadByte(*str) && *(str+1)){
  117.             highbyte = *(str);
  118.             lowbyte  = *(str+1);
  119.         }else{
  120.             highbyte = 0;
  121.             lowbyte  = *str;
  122.         }
  123.     };
  124.     bool isnull(){
  125.         return (highbyte == 0) && (lowbyte == 0);
  126.     }
  127.     operator int(){
  128.         return highbyte*256 + lowbyte;
  129.     }
  130. private:
  131.     char highbyte;
  132.     char lowbyte;
  133.     friend bool operator==(dbcschar c1, dbcschar c2);
  134. };
  135. bool operator==(dbcschar c1, dbcschar c2)
  136. {
  137.     /* ignore case comaration */
  138.     if(c1.highbyte == 0 && c2.highbyte == 0){
  139.         return tolower(c1.lowbyte) == tolower(c2.lowbyte);
  140.     }else{
  141.         return c1.highbyte == c2.highbyte && c1.lowbyte == c2.lowbyte;
  142.     }
  143. }
  144. /*
  145.     ASCII-Caracter    => 0x00cc
  146.     HANKAKU-KANA    => 0x00cc
  147.     2 byte KANJI    => 0xcccc
  148. */
  149. typedef basic_string<dbcschar> dbcsstring;
  150.  
  151. static bool is_regexp_match(dbcsstring::iterator regexp_begin, dbcsstring::iterator regexp_end
  152.                      , dbcsstring::iterator str_begin, dbcsstring::iterator str_end)
  153. {
  154.     switch(*regexp_begin){
  155.     case '*':
  156.         if(str_begin == str_end){return true;}
  157.         return is_regexp_match(regexp_begin+1, regexp_end, str_begin  , str_end) 
  158.             || is_regexp_match(regexp_begin  , regexp_end, str_begin+1, str_end);
  159.         break;
  160.     case '?':
  161.         if(str_begin == str_end){return false;}
  162.         return is_regexp_match(regexp_begin+1, regexp_end, str_begin+1, str_end);
  163.         break;
  164.     case '\0':
  165.         if(str_begin == str_end){return true;}
  166.         return false;
  167.     default:
  168.         if((*regexp_begin == '/' ||  *regexp_begin == '\\')
  169.             && (*str_begin == '/' ||  *str_begin == '\\')){
  170.             ;
  171.         }else if(*regexp_begin != *str_begin){return false;}
  172.         return is_regexp_match(regexp_begin+1, regexp_end, str_begin+1, str_end);
  173.     }
  174. }
  175. static dbcsstring str2dbcsstring(const char *str)
  176. {
  177.     dbcsstring dbcsstr;
  178.     while(*str){
  179.         dbcschar ch = str;
  180.         // ch.str2dbcschar(str);
  181.         dbcsstr.append(1, ch);
  182.         if(::IsDBCSLeadByte(*str)){
  183.             str++;
  184.         }
  185.         str++;
  186.     }
  187.     return dbcsstr;
  188. }
  189. bool is_regexp_match_dbcs(const char *regexp, const char *str)
  190. {
  191.     dbcsstring regexp_dbcsstr = str2dbcsstring(regexp);
  192.     dbcsstring str_dbcsstr = str2dbcsstring(str);
  193.     return is_regexp_match(regexp_dbcsstr.begin(), regexp_dbcsstr.end()
  194.                             , str_dbcsstr.begin(), str_dbcsstr.end());    
  195. }
  196. void convert_yen_to_slash(char *pathname)
  197. {
  198.     while(*pathname){
  199.         if(*pathname == '\\'){
  200.             *pathname = '/';
  201.         }
  202.         pathname += _mbclen((unsigned char*)pathname);
  203.     }
  204. }
  205. static void find_files(const char *regexp, list<string> &files)
  206. {
  207.     //WIN32_FIND_DATA finddata;
  208.     //HANDLE hFindFile;
  209.     long handle;
  210.     struct _finddata_t finddata;
  211.     string dirname = get_dirname(regexp);
  212.  
  213.     handle = _findfirst(regexp, &finddata);
  214.     int ret;
  215.     // hFindFile = ::FindFirstFile(regexp, &finddata);
  216.     ret = (handle != -1);
  217.     while(ret){
  218.         string fname = finddata.name;
  219.         if(fname == "." || fname == ".."){
  220.             ;
  221.         }else{
  222.             string pathname = make_pathname(dirname.c_str(), fname.c_str());
  223.             files.push_back(pathname);
  224.             if((finddata.attrib & _A_SUBDIR) == _A_SUBDIR){
  225.                 string searchname = make_pathname(pathname.c_str(), "*.*");
  226.                 find_files(searchname.c_str(), files);
  227.             }
  228.         }
  229.         ret = (_findnext(handle, &finddata) == 0);
  230.         // ret = ::FindNextFile(hFindFile, finddata);
  231.     }
  232.     if(handle != -1){
  233.         _findclose(handle);
  234.     }
  235. }
  236.  
  237. list<string> find_files(const char *regexp)
  238. {
  239.     list<string> files;
  240.     find_files(regexp, files);
  241.     return files;
  242. }
  243.