medit/moo/mooglib/moo-glib.c
2016-10-02 21:14:12 -07:00

322 lines
6.9 KiB
C

#define MOO_DO_NOT_MANGLE_GLIB_FUNCTIONS
#include "moo-glib.h"
#include "moo-time.h"
#include "moo-stat.h"
#include <sys/stat.h>
#include <sys/types.h>
#include <time.h>
#include <stdio.h>
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#ifdef __WIN32__
#include <io.h>
#endif // __WIN32__
const mgw_errno_t MGW_E_NOERROR = { MGW_ENOERROR };
const mgw_errno_t MGW_E_EXIST = { MGW_EEXIST };
static mgw_time_t convert_time_t (time_t t)
{
mgw_time_t result = { t };
return result;
}
static void convert_g_stat_buf (const GStatBuf* gbuf, MgwStatBuf* mbuf)
{
mbuf->atime = convert_time_t (gbuf->st_atime);
mbuf->mtime = convert_time_t (gbuf->st_mtime);
mbuf->ctime = convert_time_t (gbuf->st_ctime);
mbuf->size = gbuf->st_size;
mbuf->isreg = S_ISREG (gbuf->st_mode);
mbuf->isdir = S_ISDIR (gbuf->st_mode);
#ifdef S_ISLNK
mbuf->islnk = S_ISLNK (gbuf->st_mode);
#else
mbuf->islnk = 0;
#endif
#ifdef S_ISSOCK
mbuf->issock = S_ISSOCK (gbuf->st_mode);
#else
mbuf->issock = 0;
#endif
#ifdef S_ISFIFO
mbuf->isfifo = S_ISFIFO (gbuf->st_mode);
#else
mbuf->isfifo = 0;
#endif
#ifdef S_ISCHR
mbuf->ischr = S_ISCHR (gbuf->st_mode);
#else
mbuf->ischr = 0;
#endif
#ifdef S_ISBLK
mbuf->isblk = S_ISBLK (gbuf->st_mode);
#else
mbuf->isblk = 0;
#endif
}
#define call_with_errno(err__, func__, rtype__, ...) \
({ \
rtype__ result__; \
errno = 0; \
result__ = (func__) (__VA_ARGS__); \
if ((err__) != NULL) \
(err__)->value = errno; \
result__; \
})
const char *
mgw_strerror (mgw_errno_t err)
{
return g_strerror (err.value);
}
GFileError
mgw_file_error_from_errno (mgw_errno_t err)
{
return g_file_error_from_errno (err.value);
}
int
mgw_stat (const gchar *filename, MgwStatBuf *buf, mgw_errno_t *err)
{
GStatBuf gbuf = { 0 };
int result = call_with_errno (err, g_stat, int, filename, &gbuf);
convert_g_stat_buf (&gbuf, buf);
return result;
}
int
mgw_lstat (const gchar *filename, MgwStatBuf *buf, mgw_errno_t *err)
{
GStatBuf gbuf = { 0 };
int result = call_with_errno (err, g_lstat, int, filename, &gbuf);
convert_g_stat_buf (&gbuf, buf);
return result;
}
const struct tm *
mgw_localtime (const mgw_time_t *timep)
{
time_t t = timep->value;
return localtime(&t);
}
#ifdef __WIN32__
static struct tm *
localtime_r (const time_t *timep,
struct tm *result)
{
struct tm *res;
res = localtime (timep);
if (res)
*result = *res;
return res;
}
#endif
const struct tm *
mgw_localtime_r (const mgw_time_t *timep, struct tm *result, mgw_errno_t *err)
{
time_t t = timep->value;
return call_with_errno (err, localtime_r, struct tm*, &t, result);
}
mgw_time_t
mgw_time (mgw_time_t *t, mgw_errno_t *err)
{
time_t t1;
mgw_time_t result = { call_with_errno (err, time, time_t, &t1) };
if (t != NULL)
t->value = t1;
return result;
}
guint64
mgw_ascii_strtoull (const gchar *nptr, gchar **endptr, guint base, mgw_errno_t *err)
{
return call_with_errno (err, g_ascii_strtoull, guint64, nptr, endptr, base);
}
gdouble
mgw_ascii_strtod (const gchar *nptr, gchar **endptr, mgw_errno_t *err)
{
return call_with_errno (err, g_ascii_strtod, double, nptr, endptr);
}
MGW_FILE *
mgw_fopen (const char *filename, const char *mode, mgw_errno_t *err)
{
return (MGW_FILE*) call_with_errno (err, g_fopen, FILE*, filename, mode);
}
int mgw_fclose (MGW_FILE *file)
{
return fclose ((FILE*) file);
}
gsize
mgw_fread(void *ptr, gsize size, gsize nmemb, MGW_FILE *stream, mgw_errno_t *err)
{
return call_with_errno (err, fread, gsize, ptr, size, nmemb, (FILE*) stream);
}
gsize
mgw_fwrite(const void *ptr, gsize size, gsize nmemb, MGW_FILE *stream)
{
return fwrite (ptr, size, nmemb, (FILE*) stream);
}
int
mgw_ferror (MGW_FILE *file)
{
return ferror ((FILE*) file);
}
char *
mgw_fgets(char *s, int size, MGW_FILE *stream)
{
return fgets(s, size, (FILE*) stream);
}
MgwFd
mgw_open (const char *filename, int flags, int mode)
{
MgwFd fd = { g_open (filename, flags, mode) };
return fd;
}
int
mgw_close (MgwFd fd)
{
return close (fd.value);
}
gssize
mgw_write(MgwFd fd, const void *buf, gsize count)
{
return write (fd.value, buf, count);
}
int
mgw_pipe (MgwFd *fds)
{
int t[2];
#ifndef __WIN32__
int result = pipe (t);
#else
int result = _pipe(t, 4096, O_BINARY);
#endif
fds[0].value = t[0];
fds[1].value = t[1];
return result;
}
void
mgw_perror (const char *s)
{
perror (s);
}
int
mgw_unlink (const char *path, mgw_errno_t *err)
{
return call_with_errno (err, g_unlink, int, path);
}
int
mgw_remove (const char *path, mgw_errno_t *err)
{
return call_with_errno (err, g_remove, int, path);
}
int
mgw_rename (const char *oldpath, const char *newpath, mgw_errno_t *err)
{
return call_with_errno (err, g_rename, int, oldpath, newpath);
}
int
mgw_mkdir (const gchar *filename, int mode, mgw_errno_t *err)
{
return call_with_errno (err, g_mkdir, int, filename, mode);
}
int
mgw_mkdir_with_parents (const gchar *pathname, gint mode, mgw_errno_t *err)
{
return call_with_errno (err, g_mkdir_with_parents, int, pathname, mode);
}
int
mgw_access (const char *path, mgw_access_mode_t mode)
{
int gmode = F_OK;
if (mode.value & MGW_R_OK)
gmode |= R_OK;
if (mode.value & MGW_W_OK)
gmode |= W_OK;
if (mode.value & MGW_X_OK)
gmode |= X_OK;
return g_access (path, gmode);
}
gboolean
mgw_spawn_async_with_pipes (const gchar *working_directory,
gchar **argv,
gchar **envp,
GSpawnFlags flags,
GSpawnChildSetupFunc child_setup,
gpointer user_data,
GPid *child_pid,
MgwFd *standard_input,
MgwFd *standard_output,
MgwFd *standard_error,
GError **error)
{
return g_spawn_async_with_pipes (working_directory, argv, envp, flags,
child_setup, user_data, child_pid,
standard_input ? &standard_input->value : NULL,
standard_output ? &standard_output->value : NULL,
standard_error ? &standard_error->value : NULL,
error);
}
GIOChannel *
mgw_io_channel_unix_new (MgwFd fd)
{
return g_io_channel_unix_new (fd.value);
}
#ifdef __WIN32__
GIOChannel *
mgw_io_channel_win32_new_fd (MgwFd fd)
{
return g_io_channel_win32_new_fd (fd.value);
}
#endif // __WIN32__