forked from LeenkxTeam/LNXSDK
		
	
		
			
				
	
	
		
			707 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			707 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| 
 | |
| LZ4X - An optimized LZ4 compressor
 | |
| 
 | |
| Written and placed in the public domain by Ilya Muravyov
 | |
| 
 | |
| */
 | |
| 
 | |
| #ifndef _CRT_SECURE_NO_WARNINGS
 | |
| #define _CRT_SECURE_NO_WARNINGS
 | |
| #endif
 | |
| #define _CRT_DISABLE_PERFCRIT_LOCKS
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <time.h>
 | |
| 
 | |
| #define NO_UTIME
 | |
| 
 | |
| #ifndef NO_UTIME
 | |
| #  include <sys/types.h>
 | |
| #  include <sys/stat.h>
 | |
| 
 | |
| #  ifdef _MSC_VER
 | |
| #    include <sys/utime.h>
 | |
| #  else
 | |
| #    include <utime.h>
 | |
| #  endif
 | |
| #endif
 | |
| 
 | |
| #ifndef _MSC_VER
 | |
| #  define _ftelli64 ftello64
 | |
| #endif
 | |
| 
 | |
| typedef unsigned char U8;
 | |
| typedef unsigned short U16;
 | |
| typedef unsigned int U32;
 | |
| 
 | |
| //FILE* g_in;
 | |
| //FILE* g_out;
 | |
| 
 | |
| #define LZ4_MAGIC 0x184C2102
 | |
| #define BLOCK_SIZE (8<<20) // 8 MB
 | |
| #define PADDING_LITERALS 5
 | |
| 
 | |
| #define WINDOW_BITS 16
 | |
| #define WINDOW_SIZE (1<<WINDOW_BITS)
 | |
| #define WINDOW_MASK (WINDOW_SIZE-1)
 | |
| 
 | |
| #define MIN_MATCH 4
 | |
| 
 | |
| #define EXCESS (16+(BLOCK_SIZE/255))
 | |
| 
 | |
| static U8 g_buf[BLOCK_SIZE+BLOCK_SIZE+EXCESS];
 | |
| 
 | |
| #define MIN(a, b) (((a)<(b))?(a):(b))
 | |
| #define MAX(a, b) (((a)>(b))?(a):(b))
 | |
| 
 | |
| #define LOAD_16(p) (*(const U16*)(&g_buf[p]))
 | |
| #define LOAD_32(p) (*(const U32*)(&g_buf[p]))
 | |
| #define STORE_16(p, x) (*(U16*)(&g_buf[p])=(x))
 | |
| #define COPY_32(d, s) (*(U32*)(&g_buf[d])=LOAD_32(s))
 | |
| 
 | |
| #define HASH_BITS 18
 | |
| #define HASH_SIZE (1<<HASH_BITS)
 | |
| #define NIL (-1)
 | |
| 
 | |
| #define HASH_32(p) ((LOAD_32(p)*0x9E3779B9)>>(32-HASH_BITS))
 | |
| 
 | |
| static inline void wild_copy(int d, int s, int n)
 | |
| {
 | |
|   COPY_32(d, s);
 | |
|   COPY_32(d+4, s+4);
 | |
| 
 | |
|   for (int i=8; i<n; i+=8)
 | |
|   {
 | |
|     COPY_32(d+i, s+i);
 | |
|     COPY_32(d+4+i, s+4+i);
 | |
|   }
 | |
| }
 | |
| 
 | |
| #if 0
 | |
| void compress(const int max_chain)
 | |
| {
 | |
|   static int head[HASH_SIZE];
 | |
|   static int tail[WINDOW_SIZE];
 | |
| 
 | |
|   int n;
 | |
|   while ((n=fread(g_buf, 1, BLOCK_SIZE, g_in))>0)
 | |
|   {
 | |
|     for (int i=0; i<HASH_SIZE; ++i)
 | |
|       head[i]=NIL;
 | |
| 
 | |
|     int op=BLOCK_SIZE;
 | |
|     int pp=0;
 | |
| 
 | |
|     int p=0;
 | |
|     while (p<n)
 | |
|     {
 | |
|       int best_len=0;
 | |
|       int dist=0;
 | |
| 
 | |
|       const int max_match=(n-PADDING_LITERALS)-p;
 | |
|       if (max_match>=MAX(12-PADDING_LITERALS, MIN_MATCH))
 | |
|       {
 | |
|         const int limit=MAX(p-WINDOW_SIZE, NIL);
 | |
|         int chain_len=max_chain;
 | |
| 
 | |
|         int s=head[HASH_32(p)];
 | |
|         while (s>limit)
 | |
|         {
 | |
|           if (g_buf[s+best_len]==g_buf[p+best_len] && LOAD_32(s)==LOAD_32(p))
 | |
|           {
 | |
|             int len=MIN_MATCH;
 | |
|             while (len<max_match && g_buf[s+len]==g_buf[p+len])
 | |
|               ++len;
 | |
| 
 | |
|             if (len>best_len)
 | |
|             {
 | |
|               best_len=len;
 | |
|               dist=p-s;
 | |
| 
 | |
|               if (len==max_match)
 | |
|                 break;
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           if (--chain_len==0)
 | |
|             break;
 | |
| 
 | |
|           s=tail[s&WINDOW_MASK];
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (best_len>=MIN_MATCH)
 | |
|       {
 | |
|         int len=best_len-MIN_MATCH;
 | |
|         const int nib=MIN(len, 15);
 | |
| 
 | |
|         if (pp!=p)
 | |
|         {
 | |
|           const int run=p-pp;
 | |
|           if (run>=15)
 | |
|           {
 | |
|             g_buf[op++]=(15<<4)+nib;
 | |
| 
 | |
|             int j=run-15;
 | |
|             for (; j>=255; j-=255)
 | |
|               g_buf[op++]=255;
 | |
|             g_buf[op++]=j;
 | |
|           }
 | |
|           else
 | |
|             g_buf[op++]=(run<<4)+nib;
 | |
| 
 | |
|           wild_copy(op, pp, run);
 | |
|           op+=run;
 | |
|         }
 | |
|         else
 | |
|           g_buf[op++]=nib;
 | |
| 
 | |
|         STORE_16(op, dist);
 | |
|         op+=2;
 | |
| 
 | |
|         if (len>=15)
 | |
|         {
 | |
|           len-=15;
 | |
|           for (; len>=255; len-=255)
 | |
|             g_buf[op++]=255;
 | |
|           g_buf[op++]=len;
 | |
|         }
 | |
| 
 | |
|         pp=p+best_len;
 | |
| 
 | |
|         while (p<pp)
 | |
|         {
 | |
|           const U32 h=HASH_32(p);
 | |
|           tail[p&WINDOW_MASK]=head[h];
 | |
|           head[h]=p++;
 | |
|         }
 | |
|       }
 | |
|       else
 | |
|       {
 | |
|         const U32 h=HASH_32(p);
 | |
|         tail[p&WINDOW_MASK]=head[h];
 | |
|         head[h]=p++;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (pp!=p)
 | |
|     {
 | |
|       const int run=p-pp;
 | |
|       if (run>=15)
 | |
|       {
 | |
|         g_buf[op++]=15<<4;
 | |
| 
 | |
|         int j=run-15;
 | |
|         for (; j>=255; j-=255)
 | |
|           g_buf[op++]=255;
 | |
|         g_buf[op++]=j;
 | |
|       }
 | |
|       else
 | |
|         g_buf[op++]=run<<4;
 | |
| 
 | |
|       wild_copy(op, pp, run);
 | |
|       op+=run;
 | |
|     }
 | |
| 
 | |
|     const int comp_len=op-BLOCK_SIZE;
 | |
|     fwrite(&comp_len, 1, sizeof(comp_len), g_out);
 | |
|     fwrite(&g_buf[BLOCK_SIZE], 1, comp_len, g_out);
 | |
| 
 | |
|     fprintf(stderr, "%lld -> %lld\r", _ftelli64(g_in), _ftelli64(g_out));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void compress_optimal()
 | |
| {
 | |
|   static int head[HASH_SIZE];
 | |
|   static int nodes[WINDOW_SIZE][2];
 | |
|   static struct
 | |
|   {
 | |
|     int cum;
 | |
| 
 | |
|     int len;
 | |
|     int dist;
 | |
|   } path[BLOCK_SIZE+1];
 | |
| 
 | |
|   int n;
 | |
|   while ((n=fread(g_buf, 1, BLOCK_SIZE, g_in))>0)
 | |
|   {
 | |
|     // Pass 1: Find all matches
 | |
| 
 | |
|     for (int i=0; i<HASH_SIZE; ++i)
 | |
|       head[i]=NIL;
 | |
| 
 | |
|     for (int p=0; p<n; ++p)
 | |
|     {
 | |
|       int best_len=0;
 | |
|       int dist=0;
 | |
| 
 | |
|       const int max_match=(n-PADDING_LITERALS)-p;
 | |
|       if (max_match>=MAX(12-PADDING_LITERALS, MIN_MATCH))
 | |
|       {
 | |
|         const int limit=MAX(p-WINDOW_SIZE, NIL);
 | |
| 
 | |
|         int* left=&nodes[p&WINDOW_MASK][1];
 | |
|         int* right=&nodes[p&WINDOW_MASK][0];
 | |
| 
 | |
|         int left_len=0;
 | |
|         int right_len=0;
 | |
| 
 | |
|         const U32 h=HASH_32(p);
 | |
|         int s=head[h];
 | |
|         head[h]=p;
 | |
| 
 | |
|         while (s>limit)
 | |
|         {
 | |
|           int len=MIN(left_len, right_len);
 | |
| 
 | |
|           if (g_buf[s+len]==g_buf[p+len])
 | |
|           {
 | |
|             while (++len<max_match && g_buf[s+len]==g_buf[p+len]);
 | |
| 
 | |
|             if (len>best_len)
 | |
|             {
 | |
|               best_len=len;
 | |
|               dist=p-s;
 | |
| 
 | |
|               if (len==max_match || len>=(1<<16))
 | |
|                 break;
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           if (g_buf[s+len]<g_buf[p+len])
 | |
|           {
 | |
|             *right=s;
 | |
|             right=&nodes[s&WINDOW_MASK][1];
 | |
|             s=*right;
 | |
|             right_len=len;
 | |
|           }
 | |
|           else
 | |
|           {
 | |
|             *left=s;
 | |
|             left=&nodes[s&WINDOW_MASK][0];
 | |
|             s=*left;
 | |
|             left_len=len;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         *left=NIL;
 | |
|         *right=NIL;
 | |
|       }
 | |
| 
 | |
|       path[p].len=best_len;
 | |
|       path[p].dist=dist;
 | |
|     }
 | |
| 
 | |
|     // Pass 2: Build the shortest path
 | |
| 
 | |
|     path[n].cum=0;
 | |
| 
 | |
|     int count=15;
 | |
| 
 | |
|     for (int p=n-1; p>0; --p)
 | |
|     {
 | |
|       int c0=path[p+1].cum+1;
 | |
| 
 | |
|       if (--count==0)
 | |
|       {
 | |
|         count=255;
 | |
|         ++c0;
 | |
|       }
 | |
| 
 | |
|       int len=path[p].len;
 | |
|       if (len>=MIN_MATCH)
 | |
|       {
 | |
|         int c1=1<<30;
 | |
| 
 | |
|         const int j=MAX(len-255, MIN_MATCH);
 | |
|         for (int i=len; i>=j; --i)
 | |
|         {
 | |
|           int tmp=path[p+i].cum+3;
 | |
| 
 | |
|           if (i>=(15+MIN_MATCH))
 | |
|             tmp+=1+((i-(15+MIN_MATCH))/255);
 | |
| 
 | |
|           if (tmp<c1)
 | |
|           {
 | |
|             c1=tmp;
 | |
|             len=i;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         if (c1<=c0)
 | |
|         {
 | |
|           path[p].cum=c1;
 | |
|           path[p].len=len;
 | |
| 
 | |
|           count=15;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|           path[p].cum=c0;
 | |
|           path[p].len=0;
 | |
|         }
 | |
|       }
 | |
|       else
 | |
|         path[p].cum=c0;
 | |
|     }
 | |
| 
 | |
|     // Pass 3: Output the codes
 | |
| 
 | |
|     int op=BLOCK_SIZE;
 | |
|     int pp=0;
 | |
| 
 | |
|     int p=0;
 | |
|     while (p<n)
 | |
|     {
 | |
|       if (path[p].len>=MIN_MATCH)
 | |
|       {
 | |
|         int len=path[p].len-MIN_MATCH;
 | |
|         const int nib=MIN(len, 15);
 | |
| 
 | |
|         if (pp!=p)
 | |
|         {
 | |
|           const int run=p-pp;
 | |
|           if (run>=15)
 | |
|           {
 | |
|             g_buf[op++]=(15<<4)+nib;
 | |
| 
 | |
|             int j=run-15;
 | |
|             for (; j>=255; j-=255)
 | |
|               g_buf[op++]=255;
 | |
|             g_buf[op++]=j;
 | |
|           }
 | |
|           else
 | |
|             g_buf[op++]=(run<<4)+nib;
 | |
| 
 | |
|           wild_copy(op, pp, run);
 | |
|           op+=run;
 | |
|         }
 | |
|         else
 | |
|           g_buf[op++]=nib;
 | |
| 
 | |
|         STORE_16(op, path[p].dist);
 | |
|         op+=2;
 | |
| 
 | |
|         if (len>=15)
 | |
|         {
 | |
|           len-=15;
 | |
|           for (; len>=255; len-=255)
 | |
|             g_buf[op++]=255;
 | |
|           g_buf[op++]=len;
 | |
|         }
 | |
| 
 | |
|         p+=path[p].len;
 | |
| 
 | |
|         pp=p;
 | |
|       }
 | |
|       else
 | |
|         ++p;
 | |
|     }
 | |
| 
 | |
|     if (pp!=p)
 | |
|     {
 | |
|       const int run=p-pp;
 | |
|       if (run>=15)
 | |
|       {
 | |
|         g_buf[op++]=15<<4;
 | |
| 
 | |
|         int j=run-15;
 | |
|         for (; j>=255; j-=255)
 | |
|           g_buf[op++]=255;
 | |
|         g_buf[op++]=j;
 | |
|       }
 | |
|       else
 | |
|         g_buf[op++]=run<<4;
 | |
| 
 | |
|       wild_copy(op, pp, run);
 | |
|       op+=run;
 | |
|     }
 | |
| 
 | |
|     const int comp_len=op-BLOCK_SIZE;
 | |
|     fwrite(&comp_len, 1, sizeof(comp_len), g_out);
 | |
|     fwrite(&g_buf[BLOCK_SIZE], 1, comp_len, g_out);
 | |
| 
 | |
|     fprintf(stderr, "%lld -> %lld\r", _ftelli64(g_in), _ftelli64(g_out));
 | |
|   }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static size_t kread(void* dst, size_t size, const char* src, size_t* offset, size_t compressedSize) {
 | |
|   size_t realSize = MIN(size, compressedSize - *offset);
 | |
|   memcpy(dst, &src[*offset], realSize);
 | |
|   *offset += realSize;
 | |
|   return realSize;
 | |
| }
 | |
| 
 | |
| static size_t kwrite(void* src, size_t size, char* dst, size_t* offset, int maxOutputSize) {
 | |
|   size_t realSize = MIN(size, maxOutputSize - *offset);
 | |
|   memcpy(&dst[*offset], src, size);
 | |
|   *offset += realSize;
 | |
|   return realSize;
 | |
| }
 | |
| 
 | |
| //int decompress()
 | |
| #ifdef KINC_LZ4X
 | |
| #include <kinc/error.h>
 | |
| 
 | |
| int LZ4_decompress_safe(const char *source, char *buf, int compressedSize, int maxOutputSize)
 | |
| {
 | |
|   size_t read_offset = 0;
 | |
|   size_t write_offset = 0;
 | |
|   int comp_len;
 | |
|   while (kread(&comp_len, sizeof(comp_len), source, &read_offset, compressedSize)>0)
 | |
|   {
 | |
|     if (comp_len<2 || comp_len>(BLOCK_SIZE+EXCESS)
 | |
|         || kread(&g_buf[BLOCK_SIZE], comp_len, source, &read_offset, compressedSize)!=comp_len)
 | |
|       return -1;
 | |
| 
 | |
|     int p=0;
 | |
| 
 | |
|     int ip=BLOCK_SIZE;
 | |
|     const int ip_end=ip+comp_len;
 | |
| 
 | |
|     for (;;)
 | |
|     {
 | |
|       const int token=g_buf[ip++];
 | |
|       if (token>=16)
 | |
|       {
 | |
|         int run=token>>4;
 | |
|         if (run==15)
 | |
|         {
 | |
|           for (;;)
 | |
|           {
 | |
|             const int c=g_buf[ip++];
 | |
|             run+=c;
 | |
|             if (c!=255)
 | |
|               break;
 | |
|           }
 | |
|         }
 | |
|         if ((p+run)>BLOCK_SIZE)
 | |
|           return -1;
 | |
| 
 | |
|         wild_copy(p, ip, run);
 | |
|         p+=run;
 | |
|         ip+=run;
 | |
|         if (ip>=ip_end)
 | |
|           break;
 | |
|       }
 | |
| 
 | |
|       int s=p-LOAD_16(ip);
 | |
|       ip+=2;
 | |
|       if (s<0)
 | |
|         return -1;
 | |
| 
 | |
|       int len=(token&15)+MIN_MATCH;
 | |
|       if (len==(15+MIN_MATCH))
 | |
|       {
 | |
|         for (;;)
 | |
|         {
 | |
|           const int c=g_buf[ip++];
 | |
|           len+=c;
 | |
|           if (c!=255)
 | |
|             break;
 | |
|         }
 | |
|       }
 | |
|       if ((p+len)>BLOCK_SIZE)
 | |
|         return -1;
 | |
| 
 | |
|       if ((p-s)>=4)
 | |
|       {
 | |
|         wild_copy(p, s, len);
 | |
|         p+=len;
 | |
|       }
 | |
|       else
 | |
|       {
 | |
|         while (len--!=0)
 | |
|           g_buf[p++]=g_buf[s++];
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (kwrite(g_buf, p, buf, &write_offset, maxOutputSize)!=p)
 | |
|     {
 | |
|       kinc_error_message("Fwrite() failed");
 | |
|       return -1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #if 0
 | |
| int main(int argc, char** argv)
 | |
| {
 | |
|   const clock_t start=clock();
 | |
| 
 | |
|   int level=4;
 | |
|   bool do_decomp=false;
 | |
|   bool overwrite=false;
 | |
| 
 | |
|   while (argc>1 && *argv[1]=='-')
 | |
|   {
 | |
|     for (int i=1; argv[1][i]!='\0'; ++i)
 | |
|     {
 | |
|       switch (argv[1][i])
 | |
|       {
 | |
|       case '1':
 | |
|       case '2':
 | |
|       case '3':
 | |
|       case '4':
 | |
|       case '5':
 | |
|       case '6':
 | |
|       case '7':
 | |
|       case '8':
 | |
|       case '9':
 | |
|         level=argv[1][i]-'0';
 | |
|         break;
 | |
|       case 'd':
 | |
|         do_decomp=true;
 | |
|         break;
 | |
|       case 'f':
 | |
|         overwrite=true;
 | |
|         break;
 | |
|       default:
 | |
|         fprintf(stderr, "Unknown option: -%c\n", argv[1][i]);
 | |
|         exit(1);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     --argc;
 | |
|     ++argv;
 | |
|   }
 | |
| 
 | |
|   if (argc<2)
 | |
|   {
 | |
|     fprintf(stderr,
 | |
|         "LZ4X - An optimized LZ4 compressor, v1.60\n"
 | |
|         "Written and placed in the public domain by Ilya Muravyov\n"
 | |
|         "\n"
 | |
|         "Usage: LZ4X [options] infile [outfile]\n"
 | |
|         "\n"
 | |
|         "Options:\n"
 | |
|         "  -1  Compress faster\n"
 | |
|         "  -9  Compress better\n"
 | |
|         "  -d  Decompress\n"
 | |
|         "  -f  Force overwrite of output file\n");
 | |
|     exit(1);
 | |
|   }
 | |
| 
 | |
|   g_in=fopen(argv[1], "rb");
 | |
|   if (!g_in)
 | |
|   {
 | |
|     perror(argv[1]);
 | |
|     exit(1);
 | |
|   }
 | |
| 
 | |
|   char out_name[FILENAME_MAX];
 | |
|   if (argc<3)
 | |
|   {
 | |
|     strcpy(out_name, argv[1]);
 | |
|     if (do_decomp)
 | |
|     {
 | |
|       const int p=strlen(out_name)-4;
 | |
|       if (p>0 && strcmp(&out_name[p], ".lz4")==0)
 | |
|         out_name[p]='\0';
 | |
|       else
 | |
|         strcat(out_name, ".out");
 | |
|     }
 | |
|     else
 | |
|       strcat(out_name, ".lz4");
 | |
|   }
 | |
|   else
 | |
|     strcpy(out_name, argv[2]);
 | |
| 
 | |
|   if (!overwrite)
 | |
|   {
 | |
|     FILE* f=fopen(out_name, "rb");
 | |
|     if (f)
 | |
|     {
 | |
|       fclose(f);
 | |
| 
 | |
|       fprintf(stderr, "%s already exists. Overwrite (y/n)? ", out_name);
 | |
|       fflush(stderr);
 | |
| 
 | |
|       if (getchar()!='y')
 | |
|       {
 | |
|         fprintf(stderr, "Not overwritten\n");
 | |
|         exit(1);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (do_decomp)
 | |
|   {
 | |
|     int magic;
 | |
|     fread(&magic, 1, sizeof(magic), g_in);
 | |
|     if (magic!=LZ4_MAGIC)
 | |
|     {
 | |
|       fprintf(stderr, "%s: Not in Legacy format\n", argv[1]);
 | |
|       exit(1);
 | |
|     }
 | |
| 
 | |
|     g_out=fopen(out_name, "wb");
 | |
|     if (!g_out)
 | |
|     {
 | |
|       perror(out_name);
 | |
|       exit(1);
 | |
|     }
 | |
| 
 | |
|     fprintf(stderr, "Decompressing %s:\n", argv[1]);
 | |
| 
 | |
|     if (decompress()!=0)
 | |
|     {
 | |
|       fprintf(stderr, "%s: Corrupt input\n", argv[1]);
 | |
|       exit(1);
 | |
|     }
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     g_out=fopen(out_name, "wb");
 | |
|     if (!g_out)
 | |
|     {
 | |
|       perror(out_name);
 | |
|       exit(1);
 | |
|     }
 | |
| 
 | |
|     const int magic=LZ4_MAGIC;
 | |
|     fwrite(&magic, 1, sizeof(magic), g_out);
 | |
| 
 | |
|     fprintf(stderr, "Compressing %s:\n", argv[1]);
 | |
| 
 | |
|     if (level==9)
 | |
|       compress_optimal();
 | |
|     else
 | |
|       compress((level<8)?1<<level:WINDOW_SIZE);
 | |
|   }
 | |
| 
 | |
|   fprintf(stderr, "%lld -> %lld in %1.3f sec\n", _ftelli64(g_in),
 | |
|       _ftelli64(g_out), double(clock()-start)/CLOCKS_PER_SEC);
 | |
| 
 | |
|   fclose(g_in);
 | |
|   fclose(g_out);
 | |
| 
 | |
| #ifndef NO_UTIME
 | |
|   struct _stati64 sb;
 | |
|   if (_stati64(argv[1], &sb)!=0)
 | |
|   {
 | |
|     perror("Stat() failed");
 | |
|     exit(1);
 | |
|   }
 | |
|   struct utimbuf ub;
 | |
|   ub.actime=sb.st_atime;
 | |
|   ub.modtime=sb.st_mtime;
 | |
|   if (utime(out_name, &ub)!=0)
 | |
|   {
 | |
|     perror("Utime() failed");
 | |
|     exit(1);
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| #endif
 |