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
|