2017-10-23 01:12:04 -03:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
2021-12-10 08:27:01 -03:00
|
|
|
#include <stdlib.h>
|
2017-10-23 01:12:04 -03:00
|
|
|
#include "types.h"
|
|
|
|
#include "ioutil.h"
|
2021-12-08 17:14:20 -03:00
|
|
|
#include "vec.h"
|
|
|
|
#include <stdio.h>
|
2018-02-21 21:35:58 -03:00
|
|
|
|
|
|
|
#ifndef _WIN32
|
|
|
|
|
2017-10-23 01:12:04 -03:00
|
|
|
#include <errno.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
int writeall(FH fd,const u8 *data,size_t len)
|
|
|
|
{
|
|
|
|
ssize_t wrote;
|
|
|
|
while (len) {
|
|
|
|
wrote = write(fd,data,len);
|
|
|
|
if (wrote == -1) {
|
|
|
|
if (errno == EAGAIN || errno == EWOULDBLOCK || errno == EINTR)
|
|
|
|
continue;
|
|
|
|
return -1;
|
|
|
|
}
|
2018-02-22 21:11:20 -03:00
|
|
|
len -= (size_t) wrote;
|
2017-10-23 01:12:04 -03:00
|
|
|
data += wrote;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
FH createfile(const char *path,int secret)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
do {
|
|
|
|
fd = open(path,O_WRONLY | O_CREAT | O_TRUNC,secret ? 0600 : 0666);
|
2021-12-08 17:14:20 -03:00
|
|
|
if (fd < 0) {
|
2017-10-23 01:12:04 -03:00
|
|
|
if (errno == EINTR)
|
|
|
|
continue;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
int closefile(FH fd)
|
|
|
|
{
|
|
|
|
int cret;
|
|
|
|
do {
|
|
|
|
cret = close(fd);
|
2021-12-08 17:14:20 -03:00
|
|
|
if (cret < 0) {
|
2017-10-23 01:12:04 -03:00
|
|
|
if (errno == EINTR)
|
|
|
|
continue;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-02-21 21:35:58 -03:00
|
|
|
int createdir(const char *path,int secret)
|
|
|
|
{
|
|
|
|
return mkdir(path,secret ? 0700 : 0777);
|
|
|
|
}
|
|
|
|
|
2021-12-10 08:27:01 -03:00
|
|
|
static int syncwritefile(const char *filename,const char *tmpname,int secret,const u8 *data,size_t datalen)
|
2021-12-08 17:14:20 -03:00
|
|
|
{
|
2021-12-10 08:27:01 -03:00
|
|
|
FH f = createfile(tmpname,secret);
|
2021-12-08 17:14:20 -03:00
|
|
|
if (f == FH_invalid)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (writeall(f,data,datalen) < 0) {
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failclose;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
int sret;
|
|
|
|
do {
|
|
|
|
sret = fsync(f);
|
|
|
|
if (sret < 0) {
|
|
|
|
if (errno == EINTR)
|
|
|
|
continue;
|
|
|
|
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failclose;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
if (closefile(f) < 0) {
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failrm;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
2021-12-10 08:27:01 -03:00
|
|
|
if (rename(tmpname,filename) < 0) {
|
|
|
|
goto failrm;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
2021-12-10 08:27:01 -03:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
failclose:
|
|
|
|
(void) closefile(f);
|
|
|
|
failrm:
|
|
|
|
remove(tmpname);
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int syncwrite(const char *filename,int secret,const u8 *data,size_t datalen)
|
|
|
|
{
|
|
|
|
//fprintf(stderr,"filename = %s\n",filename);
|
|
|
|
|
|
|
|
size_t fnlen = strlen(filename);
|
|
|
|
|
|
|
|
VEC_STRUCT(,char) tmpnamebuf;
|
|
|
|
VEC_INIT(tmpnamebuf);
|
|
|
|
VEC_ADDN(tmpnamebuf,fnlen + 4 /* ".tmp" */ + 1 /* "\0" */);
|
|
|
|
memcpy(&VEC_BUF(tmpnamebuf,0),filename,fnlen);
|
|
|
|
strcpy(&VEC_BUF(tmpnamebuf,fnlen),".tmp");
|
|
|
|
const char *tmpname = &VEC_BUF(tmpnamebuf,0);
|
|
|
|
|
|
|
|
//fprintf(stderr,"tmpname = %s\n",tmpname);
|
|
|
|
|
|
|
|
int r = syncwritefile(filename,tmpname,secret,data,datalen);
|
|
|
|
|
|
|
|
VEC_FREE(tmpnamebuf);
|
|
|
|
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
|
|
|
|
VEC_STRUCT(,char) dirnamebuf;
|
|
|
|
VEC_INIT(dirnamebuf);
|
|
|
|
const char *dirname;
|
2021-12-08 17:14:20 -03:00
|
|
|
|
|
|
|
for (ssize_t x = ((ssize_t)fnlen) - 1;x >= 0;--x) {
|
|
|
|
if (filename[x] == '/') {
|
|
|
|
if (x)
|
|
|
|
--x;
|
|
|
|
++x;
|
2021-12-10 08:27:01 -03:00
|
|
|
VEC_ADDN(dirnamebuf,x + 1);
|
|
|
|
memcpy(&VEC_BUF(dirnamebuf,0),filename,x);
|
|
|
|
VEC_BUF(dirnamebuf,x) = '\0';
|
|
|
|
dirname = &VEC_BUF(dirnamebuf,0);
|
2021-12-08 17:14:20 -03:00
|
|
|
goto foundslash;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* not found slash, fall back to "." */
|
2021-12-10 08:27:01 -03:00
|
|
|
dirname = ".";
|
2021-12-08 17:14:20 -03:00
|
|
|
|
|
|
|
foundslash:
|
2021-12-10 08:27:01 -03:00
|
|
|
//fprintf(stderr,"dirname = %s\n",dirname);
|
2021-12-08 17:14:20 -03:00
|
|
|
;
|
|
|
|
|
|
|
|
int dirf;
|
|
|
|
do {
|
2021-12-10 08:27:01 -03:00
|
|
|
dirf = open(dirname,O_RDONLY);
|
2021-12-08 17:14:20 -03:00
|
|
|
if (dirf < 0) {
|
|
|
|
if (errno == EINTR)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// failed for non-eintr reasons
|
|
|
|
goto skipdsync; // don't really care enough
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
|
2021-12-10 08:27:01 -03:00
|
|
|
int sret;
|
2021-12-08 17:14:20 -03:00
|
|
|
do {
|
|
|
|
sret = fsync(dirf);
|
|
|
|
if (sret < 0) {
|
|
|
|
if (errno == EINTR)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// failed for non-eintr reasons
|
|
|
|
break; // don't care
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
(void) closefile(dirf); // don't care
|
|
|
|
|
|
|
|
skipdsync:
|
2021-12-10 08:27:01 -03:00
|
|
|
VEC_FREE(dirnamebuf);
|
2021-12-08 17:14:20 -03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-02-21 21:35:58 -03:00
|
|
|
#else
|
|
|
|
|
|
|
|
int writeall(FH fd,const u8 *data,size_t len)
|
|
|
|
{
|
|
|
|
DWORD wrote;
|
|
|
|
BOOL success;
|
|
|
|
while (len) {
|
|
|
|
success = WriteFile(fd,data,
|
|
|
|
len <= (DWORD)-1 ? (DWORD)len : (DWORD)-1,&wrote,0);
|
|
|
|
if (!success)
|
|
|
|
return -1;
|
|
|
|
data += wrote;
|
|
|
|
if (len >= wrote)
|
|
|
|
len -= wrote;
|
|
|
|
else
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
FH createfile(const char *path,int secret)
|
|
|
|
{
|
|
|
|
// XXX no support for non-ascii chars
|
|
|
|
// XXX don't know how to handle secret argument
|
|
|
|
(void) secret;
|
|
|
|
return CreateFileA(path,GENERIC_WRITE,0,0,CREATE_ALWAYS,0,0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int closefile(FH fd)
|
|
|
|
{
|
|
|
|
return CloseHandle(fd) ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int createdir(const char *path,int secret)
|
|
|
|
{
|
|
|
|
// XXX don't know how to handle secret argument
|
2020-05-22 13:55:37 -04:00
|
|
|
(void) secret;
|
2018-02-21 21:35:58 -03:00
|
|
|
return CreateDirectoryA(path,0) ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
2022-05-16 17:17:50 -04:00
|
|
|
static int syncwritefile(const char *filename,const char *tmpname,int secret,const u8 *data,size_t datalen)
|
2021-12-08 17:14:20 -03:00
|
|
|
{
|
2022-05-16 17:17:50 -04:00
|
|
|
FH f = createfile(tmpname,secret);
|
2022-06-24 14:08:51 -04:00
|
|
|
if (f == FH_invalid) {
|
|
|
|
//fprintf(stderr,"!failed to create\n");
|
2021-12-08 17:14:20 -03:00
|
|
|
return -1;
|
2022-06-24 14:08:51 -04:00
|
|
|
}
|
|
|
|
|
2021-12-08 17:14:20 -03:00
|
|
|
|
|
|
|
if (writeall(f,data,datalen) < 0) {
|
2022-06-24 14:08:51 -04:00
|
|
|
//fprintf(stderr,"!failed to write\n");
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failclose;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (FlushFileBuffers(f) == 0) {
|
2022-06-24 14:08:51 -04:00
|
|
|
//fprintf(stderr,"!failed to flush\n");
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failclose;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (closefile(f) < 0) {
|
2022-06-24 14:08:51 -04:00
|
|
|
//fprintf(stderr,"!failed to close\n");
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failrm;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
2022-06-24 14:08:51 -04:00
|
|
|
if (MoveFileExA(tmpname,filename,MOVEFILE_REPLACE_EXISTING) == 0) {
|
|
|
|
//fprintf(stderr,"!failed to move\n");
|
2021-12-10 08:27:01 -03:00
|
|
|
goto failrm;
|
2021-12-08 17:14:20 -03:00
|
|
|
}
|
|
|
|
|
2021-12-10 08:27:01 -03:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
failclose:
|
|
|
|
(void) closefile(f);
|
|
|
|
failrm:
|
2022-05-16 17:17:50 -04:00
|
|
|
remove(tmpname);
|
2021-12-10 08:27:01 -03:00
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2022-05-16 17:17:50 -04:00
|
|
|
int syncwrite(const char *filename,int secret,const u8 *data,size_t datalen)
|
2021-12-10 08:27:01 -03:00
|
|
|
{
|
|
|
|
size_t fnlen = strlen(filename);
|
|
|
|
|
|
|
|
VEC_STRUCT(,char) tmpnamebuf;
|
|
|
|
VEC_INIT(tmpnamebuf);
|
|
|
|
VEC_ADDN(tmpnamebuf,fnlen + 4 /* ".tmp" */ + 1 /* "\0" */);
|
|
|
|
memcpy(&VEC_BUF(tmpnamebuf,0),filename,fnlen);
|
|
|
|
strcpy(&VEC_BUF(tmpnamebuf,fnlen),".tmp");
|
|
|
|
const char *tmpname = &VEC_BUF(tmpnamebuf,0);
|
|
|
|
|
|
|
|
int r = syncwritefile(filename,tmpname,secret,data,datalen);
|
|
|
|
|
|
|
|
VEC_FREE(tmpnamebuf);
|
|
|
|
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
|
2021-12-08 17:14:20 -03:00
|
|
|
// can't fsync parent dir on windows so just end here
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-02-21 21:35:58 -03:00
|
|
|
#endif
|
|
|
|
|
2017-10-23 01:12:04 -03:00
|
|
|
int writetofile(const char *path,const u8 *data,size_t len,int secret)
|
|
|
|
{
|
|
|
|
FH fd = createfile(path,secret);
|
|
|
|
int wret = writeall(fd,data,len);
|
|
|
|
int cret = closefile(fd);
|
|
|
|
if (cret == -1)
|
|
|
|
return -1;
|
|
|
|
return wret;
|
|
|
|
}
|