xrdp/common/os_calls.c

1343 lines
27 KiB
C
Raw Normal View History

2005-01-07 03:56:38 +03:00
/*
2007-01-12 08:01:58 +03:00
Copyright (c) 2004-2007 Jay Sorg
2006-10-12 08:46:40 +04:00
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
2005-01-07 03:56:38 +03:00
generic operating system calls
put all the os / arch define in here you want
2005-01-07 03:56:38 +03:00
*/
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
#include <windows.h>
#include <winsock.h>
#else
/* fix for solaris 10 with gcc 3.3.2 problem */
#if defined(sun) || defined(__sun)
#define ctid_t id_t
#endif
2005-01-07 03:56:38 +03:00
#include <unistd.h>
#include <errno.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <sys/time.h>
#include <sys/types.h>
2005-07-10 04:25:31 +04:00
#include <sys/wait.h>
#include <sys/stat.h>
2005-01-07 03:56:38 +03:00
#include <dlfcn.h>
#include <arpa/inet.h>
#include <netdb.h>
2005-02-08 06:45:30 +03:00
#include <signal.h>
#include <fcntl.h>
2005-07-10 04:25:31 +04:00
#include <pwd.h>
2005-12-10 05:26:26 +03:00
#include <time.h>
#include <grp.h>
2005-01-07 03:56:38 +03:00
#endif
2005-01-07 03:56:38 +03:00
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include <stdio.h>
#include "os_calls.h"
#include "arch.h"
2005-10-11 08:35:07 +04:00
/* for clearenv() */
#if defined(_WIN32)
#else
extern char** environ;
#endif
/* for solaris */
#if !defined(PF_LOCAL)
#define PF_LOCAL AF_UNIX
#endif
#if !defined(INADDR_NONE)
#define INADDR_NONE ((unsigned long)-1)
#endif
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
2006-06-18 08:59:19 +04:00
/* allocate memory, returns a pointer to it, size bytes are allocated,
if zero is non zero, each byte will be set to zero */
void* APP_CC
2005-06-28 07:04:36 +04:00
g_malloc(int size, int zero)
2005-01-07 03:56:38 +03:00
{
char* rv;
rv = (char*)malloc(size);
if (zero)
{
2005-01-07 03:56:38 +03:00
memset(rv, 0, size);
}
2005-01-07 03:56:38 +03:00
return rv;
}
/*****************************************************************************/
2006-06-18 08:59:19 +04:00
/* free the memory pointed to by ptr, ptr can be zero */
void APP_CC
2005-06-28 07:04:36 +04:00
g_free(void* ptr)
2005-01-07 03:56:38 +03:00
{
if (ptr != 0)
{
free(ptr);
}
}
/*****************************************************************************/
2006-06-18 08:59:19 +04:00
/* output text to stdout, try to use g_write / g_writeln instead to avoid
linux / windows EOL problems */
void DEFAULT_CC
g_printf(const char* format, ...)
2005-01-07 03:56:38 +03:00
{
va_list ap;
va_start(ap, format);
vfprintf(stdout, format, ap);
va_end(ap);
}
/*****************************************************************************/
void DEFAULT_CC
g_sprintf(char* dest, const char* format, ...)
2005-01-07 03:56:38 +03:00
{
va_list ap;
va_start(ap, format);
vsprintf(dest, format, ap);
va_end(ap);
}
2006-08-23 07:44:30 +04:00
/*****************************************************************************/
void DEFAULT_CC
g_snprintf(char* dest, int len, const char* format, ...)
2006-08-23 07:44:30 +04:00
{
va_list ap;
va_start(ap, format);
vsnprintf(dest, len, format, ap);
va_end(ap);
}
2006-03-21 05:05:38 +03:00
/*****************************************************************************/
void DEFAULT_CC
g_writeln(const char* format, ...)
2006-03-21 05:05:38 +03:00
{
va_list ap;
va_start(ap, format);
vfprintf(stdout, format, ap);
va_end(ap);
#if defined(_WIN32)
g_printf("\r\n");
#else
g_printf("\n");
#endif
}
/*****************************************************************************/
void DEFAULT_CC
g_write(const char* format, ...)
2006-03-21 05:05:38 +03:00
{
va_list ap;
va_start(ap, format);
vfprintf(stdout, format, ap);
va_end(ap);
}
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
/* produce a hex dump */
void APP_CC
2005-06-28 07:04:36 +04:00
g_hexdump(char* p, int len)
2005-01-07 03:56:38 +03:00
{
unsigned char* line;
int i;
int thisline;
int offset;
line = (unsigned char*)p;
offset = 0;
while (offset < len)
{
2005-02-08 06:45:30 +03:00
g_printf("%04x ", offset);
2005-01-07 03:56:38 +03:00
thisline = len - offset;
if (thisline > 16)
{
2005-01-07 03:56:38 +03:00
thisline = 16;
}
2005-01-07 03:56:38 +03:00
for (i = 0; i < thisline; i++)
{
2005-02-08 06:45:30 +03:00
g_printf("%02x ", line[i]);
}
2005-01-07 03:56:38 +03:00
for (; i < 16; i++)
{
2005-02-08 06:45:30 +03:00
g_printf(" ");
}
2005-01-07 03:56:38 +03:00
for (i = 0; i < thisline; i++)
{
2005-02-08 06:45:30 +03:00
g_printf("%c", (line[i] >= 0x20 && line[i] < 0x7f) ? line[i] : '.');
}
g_writeln("");
2005-01-07 03:56:38 +03:00
offset += thisline;
line += thisline;
}
}
/*****************************************************************************/
void APP_CC
2005-06-28 07:04:36 +04:00
g_memset(void* ptr, int val, int size)
2005-01-07 03:56:38 +03:00
{
memset(ptr, val, size);
}
/*****************************************************************************/
void APP_CC
2005-06-28 07:04:36 +04:00
g_memcpy(void* d_ptr, const void* s_ptr, int size)
2005-01-07 03:56:38 +03:00
{
memcpy(d_ptr, s_ptr, size);
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_getchar(void)
2005-01-07 03:56:38 +03:00
{
return getchar();
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_set_no_delay(int sck)
{
int i;
i = 1;
2005-11-07 03:04:11 +03:00
#if defined(_WIN32)
setsockopt(sck, IPPROTO_TCP, TCP_NODELAY, (char*)&i, sizeof(i));
#else
setsockopt(sck, IPPROTO_TCP, TCP_NODELAY, (void*)&i, sizeof(i));
2005-11-07 03:04:11 +03:00
#endif
return 0;
}
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_socket(void)
2005-01-07 03:56:38 +03:00
{
int rv;
int i;
rv = socket(PF_INET, SOCK_STREAM, 0);
2005-11-07 03:04:11 +03:00
#if defined(_WIN32)
2006-10-04 07:48:58 +04:00
i = 1;
2005-11-07 03:04:11 +03:00
setsockopt(rv, IPPROTO_TCP, TCP_NODELAY, (char*)&i, sizeof(i));
2006-10-04 07:48:58 +04:00
i = 1;
setsockopt(rv, SOL_SOCKET, SO_REUSEADDR, (char*)&i, sizeof(i));
i = 8192 * 2;
setsockopt(rv, SOL_SOCKET, SO_SNDBUF, (char*)&i, sizeof(i));
2005-11-07 03:04:11 +03:00
#else
2006-10-04 07:48:58 +04:00
i = 1;
2005-01-07 03:56:38 +03:00
setsockopt(rv, IPPROTO_TCP, TCP_NODELAY, (void*)&i, sizeof(i));
2006-10-04 07:48:58 +04:00
i = 1;
setsockopt(rv, SOL_SOCKET, SO_REUSEADDR, (void*)&i, sizeof(i));
i = 8192 * 2;
setsockopt(rv, SOL_SOCKET, SO_SNDBUF, (void*)&i, sizeof(i));
2005-11-07 03:04:11 +03:00
#endif
2005-01-07 03:56:38 +03:00
return rv;
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_local_socket(void)
2005-01-07 03:56:38 +03:00
{
2005-03-11 05:23:30 +03:00
#if defined(_WIN32)
return 0;
#else
return socket(PF_LOCAL, SOCK_STREAM, 0);
#endif
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
void APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_close(int sck)
2005-01-07 03:56:38 +03:00
{
if (sck == 0)
{
2005-01-07 03:56:38 +03:00
return;
}
2005-03-13 04:35:55 +03:00
shutdown(sck, 2);
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
closesocket(sck);
#else
close(sck);
#endif
}
/*****************************************************************************/
int APP_CC
g_tcp_connect(int sck, const char* address, const char* port)
2005-01-07 03:56:38 +03:00
{
struct sockaddr_in s;
struct hostent* h;
g_memset(&s, 0, sizeof(struct sockaddr_in));
s.sin_family = AF_INET;
s.sin_port = htons(atoi(port));
s.sin_addr.s_addr = inet_addr(address);
if (s.sin_addr.s_addr == INADDR_NONE)
{
h = gethostbyname(address);
if (h != 0)
{
2005-01-07 03:56:38 +03:00
if (h->h_name != 0)
{
2005-01-07 03:56:38 +03:00
if (h->h_addr_list != 0)
{
2005-01-07 03:56:38 +03:00
if ((*(h->h_addr_list)) != 0)
{
2005-01-07 03:56:38 +03:00
s.sin_addr.s_addr = *((int*)(*(h->h_addr_list)));
}
}
}
}
2005-01-07 03:56:38 +03:00
}
return connect(sck, (struct sockaddr*)&s, sizeof(struct sockaddr_in));
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_set_non_blocking(int sck)
2005-01-07 03:56:38 +03:00
{
unsigned long i;
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
i = 1;
ioctlsocket(sck, FIONBIO, &i);
#else
i = fcntl(sck, F_GETFL);
i = i | O_NONBLOCK;
fcntl(sck, F_SETFL, i);
#endif
return 0;
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_bind(int sck, char* port)
2005-01-07 03:56:38 +03:00
{
struct sockaddr_in s;
memset(&s, 0, sizeof(struct sockaddr_in));
s.sin_family = AF_INET;
s.sin_port = htons(atoi(port));
s.sin_addr.s_addr = INADDR_ANY;
return bind(sck, (struct sockaddr*)&s, sizeof(struct sockaddr_in));
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_local_bind(int sck, char* port)
2005-01-07 03:56:38 +03:00
{
2005-03-11 05:23:30 +03:00
#if defined(_WIN32)
return -1;
#else
2005-01-07 03:56:38 +03:00
struct sockaddr_un s;
memset(&s, 0, sizeof(struct sockaddr_un));
s.sun_family = AF_UNIX;
strcpy(s.sun_path, port);
return bind(sck, (struct sockaddr*)&s, sizeof(struct sockaddr_un));
2005-03-11 05:23:30 +03:00
#endif
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_listen(int sck)
2005-01-07 03:56:38 +03:00
{
return listen(sck, 2);
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_accept(int sck)
2005-01-07 03:56:38 +03:00
{
struct sockaddr_in s;
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
signed int i;
#else
unsigned int i;
#endif
i = sizeof(struct sockaddr_in);
memset(&s, 0, i);
return accept(sck, (struct sockaddr*)&s, &i);
}
/*****************************************************************************/
void APP_CC
2005-06-28 07:04:36 +04:00
g_sleep(int msecs)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
Sleep(msecs);
#else
usleep(msecs * 1000);
#endif
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_last_error_would_block(int sck)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
return WSAGetLastError() == WSAEWOULDBLOCK;
#else
return (errno == EWOULDBLOCK) || (errno == EINPROGRESS);
2005-01-07 03:56:38 +03:00
#endif
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_recv(int sck, void* ptr, int len, int flags)
2005-01-07 03:56:38 +03:00
{
2005-11-07 03:04:11 +03:00
#if defined(_WIN32)
return recv(sck, (char*)ptr, len, flags);
#else
2005-01-07 03:56:38 +03:00
return recv(sck, ptr, len, flags);
2005-11-07 03:04:11 +03:00
#endif
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
int APP_CC
g_tcp_send(int sck, const void* ptr, int len, int flags)
2005-01-07 03:56:38 +03:00
{
2005-11-07 03:04:11 +03:00
#if defined(_WIN32)
return send(sck, (const char*)ptr, len, flags);
2005-11-07 03:04:11 +03:00
#else
2005-01-07 03:56:38 +03:00
return send(sck, ptr, len, flags);
2005-11-07 03:04:11 +03:00
#endif
2005-01-07 03:56:38 +03:00
}
2006-12-09 23:28:41 +03:00
/*****************************************************************************/
/* wait 'millis' milliseconds for the socket to be able to write */
/* returns boolean */
int APP_CC
2006-12-09 23:28:41 +03:00
g_tcp_can_send(int sck, int millis)
{
fd_set wfds;
struct timeval time;
int rv;
time.tv_sec = millis / 1000;
time.tv_usec = (millis * 1000) % 1000000;
FD_ZERO(&wfds);
if (sck > 0)
{
FD_SET(((unsigned int)sck), &wfds);
rv = select(sck + 1, 0, &wfds, 0, &time);
if (rv > 0)
{
return 1;
}
}
return 0;
}
2006-12-09 23:53:48 +03:00
/*****************************************************************************/
/* wait 'millis' milliseconds for the socket to be able to receive */
/* returns boolean */
int APP_CC
2006-12-09 23:53:48 +03:00
g_tcp_can_recv(int sck, int millis)
{
fd_set rfds;
struct timeval time;
int rv;
time.tv_sec = millis / 1000;
time.tv_usec = (millis * 1000) % 1000000;
FD_ZERO(&rfds);
if (sck > 0)
{
FD_SET(((unsigned int)sck), &rfds);
rv = select(sck + 1, &rfds, 0, 0, &time);
if (rv > 0)
{
return 1;
}
}
return 0;
}
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_tcp_select(int sck1, int sck2)
2005-01-07 03:56:38 +03:00
{
fd_set rfds;
struct timeval time;
int max;
int rv;
time.tv_sec = 0;
time.tv_usec = 0;
FD_ZERO(&rfds);
2005-02-08 06:45:30 +03:00
if (sck1 > 0)
{
FD_SET(((unsigned int)sck1), &rfds);
}
if (sck2 > 0)
{
FD_SET(((unsigned int)sck2), &rfds);
}
2005-01-07 03:56:38 +03:00
max = sck1;
if (sck2 > max)
{
2005-01-07 03:56:38 +03:00
max = sck2;
}
2005-01-07 03:56:38 +03:00
rv = select(max + 1, &rfds, 0, 0, &time);
if (rv > 0)
{
rv = 0;
if (FD_ISSET(((unsigned int)sck1), &rfds))
{
2005-01-07 03:56:38 +03:00
rv = rv | 1;
}
2005-01-07 03:56:38 +03:00
if (FD_ISSET(((unsigned int)sck2), &rfds))
{
2005-01-07 03:56:38 +03:00
rv = rv | 2;
}
2005-01-07 03:56:38 +03:00
}
2006-03-25 20:12:15 +03:00
else
{
rv = 0;
}
2005-01-07 03:56:38 +03:00
return rv;
}
/*****************************************************************************/
void APP_CC
2005-06-28 07:04:36 +04:00
g_random(char* data, int len)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
memset(data, 0x44, len);
#else
int fd;
memset(data, 0x44, len);
fd = open("/dev/urandom", O_RDONLY);
if (fd == -1)
{
2005-01-07 03:56:38 +03:00
fd = open("/dev/random", O_RDONLY);
}
2005-01-07 03:56:38 +03:00
if (fd != -1)
{
read(fd, data, len);
close(fd);
}
#endif
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_abs(int i)
2005-01-07 03:56:38 +03:00
{
return abs(i);
}
/*****************************************************************************/
int APP_CC
g_memcmp(const void* s1, const void* s2, int len)
2005-01-07 03:56:38 +03:00
{
return memcmp(s1, s2, len);
}
/*****************************************************************************/
2005-09-25 08:08:26 +04:00
/* returns -1 on error, else return handle or file descriptor */
int APP_CC
g_file_open(const char* file_name)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
return (int)CreateFile(file_name, GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
0, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
#else
int rv;
rv = open(file_name, O_RDWR | O_CREAT, S_IRUSR | S_IWUSR);
if (rv == -1)
{
/* can't open read / write, try to open read only */
rv = open(file_name, O_RDONLY);
}
return rv;
2005-01-07 03:56:38 +03:00
#endif
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns error, always 0 */
int APP_CC
2005-06-28 07:04:36 +04:00
g_file_close(int fd)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
CloseHandle((HANDLE)fd);
#else
close(fd);
#endif
return 0;
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* read from file, returns the number of bytes read or -1 on error */
int APP_CC
2005-06-28 07:04:36 +04:00
g_file_read(int fd, char* ptr, int len)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
if (ReadFile((HANDLE)fd, (LPVOID)ptr, (DWORD)len, (LPDWORD)&len, 0))
{
2005-01-07 03:56:38 +03:00
return len;
}
2005-01-07 03:56:38 +03:00
else
{
2005-01-07 03:56:38 +03:00
return -1;
}
2005-01-07 03:56:38 +03:00
#else
return read(fd, ptr, len);
#endif
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* write to file, returns the number of bytes writen or -1 on error */
int APP_CC
2005-06-28 07:04:36 +04:00
g_file_write(int fd, char* ptr, int len)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
if (WriteFile((HANDLE)fd, (LPVOID)ptr, (DWORD)len, (LPDWORD)&len, 0))
{
2005-01-07 03:56:38 +03:00
return len;
}
2005-01-07 03:56:38 +03:00
else
{
2005-01-07 03:56:38 +03:00
return -1;
}
2005-01-07 03:56:38 +03:00
#else
return write(fd, ptr, len);
#endif
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* move file pointer, returns offset on success, -1 on failure */
int APP_CC
2005-06-28 07:04:36 +04:00
g_file_seek(int fd, int offset)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2006-08-23 07:44:30 +04:00
int rv;
rv = (int)SetFilePointer((HANDLE)fd, offset, 0, FILE_BEGIN);
if (rv == (int)INVALID_SET_FILE_POINTER)
{
return -1;
}
else
{
return rv;
}
2005-01-07 03:56:38 +03:00
#else
2006-08-23 07:44:30 +04:00
return (int)lseek(fd, offset, SEEK_SET);
2005-01-07 03:56:38 +03:00
#endif
}
/*****************************************************************************/
/* do a write lock on a file */
/* return boolean */
int APP_CC
2005-06-28 07:04:36 +04:00
g_file_lock(int fd, int start, int len)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-01-07 03:56:38 +03:00
return LockFile((HANDLE)fd, start, 0, len, 0);
#else
struct flock lock;
lock.l_type = F_WRLCK;
lock.l_whence = SEEK_SET;
lock.l_start = start;
lock.l_len = len;
if (fcntl(fd, F_SETLK, &lock) == -1)
{
2005-01-07 03:56:38 +03:00
return 0;
}
2005-01-07 03:56:38 +03:00
return 1;
#endif
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns error, always zero */
int APP_CC
g_set_file_rights(const char* filename, int read, int write)
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
2006-08-23 07:44:30 +04:00
return 0;
2005-07-10 04:25:31 +04:00
#else
int flags;
flags = read ? S_IRUSR : 0;
flags |= write ? S_IWUSR : 0;
chmod(filename, flags);
return 0;
2006-08-23 07:44:30 +04:00
#endif
2005-07-10 04:25:31 +04:00
}
/*****************************************************************************/
/* returns error */
int APP_CC
g_chmod_hex(const char* filename, int flags)
{
#if defined(_WIN32)
return 0;
#else
2006-10-28 07:28:59 +04:00
int fl;
fl = 0;
fl |= (flags & 0x4000) ? S_ISUID : 0;
fl |= (flags & 0x2000) ? S_ISGID : 0;
fl |= (flags & 0x1000) ? S_ISVTX : 0;
fl |= (flags & 0x0400) ? S_IRUSR : 0;
fl |= (flags & 0x0200) ? S_IWUSR : 0;
fl |= (flags & 0x0100) ? S_IXUSR : 0;
fl |= (flags & 0x0040) ? S_IRGRP : 0;
fl |= (flags & 0x0020) ? S_IWGRP : 0;
fl |= (flags & 0x0010) ? S_IXGRP : 0;
fl |= (flags & 0x0004) ? S_IROTH : 0;
fl |= (flags & 0x0002) ? S_IWOTH : 0;
fl |= (flags & 0x0001) ? S_IXOTH : 0;
return chmod(filename, fl);
#endif
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns error, always zero */
int APP_CC
g_mkdir(const char* dirname)
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
2006-08-23 07:44:30 +04:00
return 0;
2005-07-10 04:25:31 +04:00
#else
mkdir(dirname, S_IRWXU);
return 0;
2006-08-23 07:44:30 +04:00
#endif
2005-07-10 04:25:31 +04:00
}
/*****************************************************************************/
2006-06-18 08:59:19 +04:00
/* gets the current working directory and puts up to maxlen chars in
dirname
always returns 0 */
char* APP_CC
2005-07-10 04:25:31 +04:00
g_get_current_dir(char* dirname, int maxlen)
{
#if defined(_WIN32)
2006-06-18 08:59:19 +04:00
GetCurrentDirectory(maxlen, dirname);
return 0;
2005-07-10 04:25:31 +04:00
#else
2006-06-18 08:59:19 +04:00
getcwd(dirname, maxlen);
return 0;
2005-07-10 04:25:31 +04:00
#endif
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns error, zero on success and -1 on failure */
int APP_CC
2005-07-10 04:25:31 +04:00
g_set_current_dir(char* dirname)
{
#if defined(_WIN32)
2006-08-23 07:44:30 +04:00
if (SetCurrentDirectory(dirname))
{
return 0;
}
else
{
return -1;
}
2005-07-10 04:25:31 +04:00
#else
return chdir(dirname);
#endif
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns boolean, non zero if the file exists */
int APP_CC
g_file_exist(const char* filename)
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
return 0; // use FileAge(filename) <> -1
2005-07-10 04:25:31 +04:00
#else
return access(filename, F_OK) == 0;
#endif
}
/*****************************************************************************/
/* returns boolean, non zero if the directory exists */
int APP_CC
g_directory_exist(const char* dirname)
{
#if defined(_WIN32)
return 0; // use GetFileAttributes and check return value
// is not -1 and FILE_ATTRIBUT_DIRECTORY bit is set
#else
struct stat st;
if (stat(dirname, &st) == 0)
{
return S_ISDIR(st.st_mode);
}
else
{
return 0;
}
#endif
}
/*****************************************************************************/
/* returns boolean */
int APP_CC
g_create_dir(const char* dirname)
{
#if defined(_WIN32)
return CreateDirectory(dirname, 0); // test this
#else
return mkdir(dirname, (mode_t)-1) == 0;
#endif
}
/*****************************************************************************/
/* returns boolean */
int APP_CC
g_remove_dir(const char* dirname)
{
#if defined(_WIN32)
return RemoveDirectory(dirname); // test this
#else
return rmdir(dirname) == 0;
#endif
}
2005-11-06 02:51:10 +03:00
/*****************************************************************************/
/* returns non zero if the file was deleted */
int APP_CC
g_file_delete(const char* filename)
2005-11-06 02:51:10 +03:00
{
#if defined(_WIN32)
return DeleteFile(filename);
#else
return unlink(filename) != -1;
#endif
}
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns length of text */
int APP_CC
g_strlen(const char* text)
2005-01-07 03:56:38 +03:00
{
if (text == 0)
{
return 0;
}
2005-01-07 03:56:38 +03:00
return strlen(text);
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns dest */
char* APP_CC
g_strcpy(char* dest, const char* src)
2005-01-07 03:56:38 +03:00
{
if (src == 0 && dest != 0)
{
dest[0] = 0;
return dest;
}
if (dest == 0 || src == 0)
{
return 0;
}
2005-01-07 03:56:38 +03:00
return strcpy(dest, src);
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns dest */
char* APP_CC
g_strncpy(char* dest, const char* src, int len)
2005-01-07 03:56:38 +03:00
{
char* rv;
if (src == 0 && dest != 0)
{
dest[0] = 0;
return dest;
}
if (dest == 0 || src == 0)
{
return 0;
}
2005-01-07 03:56:38 +03:00
rv = strncpy(dest, src, len);
dest[len] = 0;
return rv;
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* returns dest */
char* APP_CC
g_strcat(char* dest, const char* src)
2005-01-07 03:56:38 +03:00
{
if (dest == 0 || src == 0)
{
return dest;
}
2005-01-07 03:56:38 +03:00
return strcat(dest, src);
}
/*****************************************************************************/
2006-08-23 07:44:30 +04:00
/* if in = 0, return 0 else return newly alloced copy of in */
char* APP_CC
g_strdup(const char* in)
2005-01-07 03:56:38 +03:00
{
int len;
char* p;
if (in == 0)
{
2005-01-07 03:56:38 +03:00
return 0;
}
2005-01-07 03:56:38 +03:00
len = g_strlen(in);
p = (char*)g_malloc(len + 1, 0);
g_strcpy(p, in);
return p;
}
2006-01-12 01:04:08 +03:00
/*****************************************************************************/
int APP_CC
g_strcmp(const char* c1, const char* c2)
2006-01-12 01:04:08 +03:00
{
return strcmp(c1, c2);
}
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
int APP_CC
g_strncmp(const char* c1, const char* c2, int len)
2005-01-07 03:56:38 +03:00
{
2005-09-25 08:08:26 +04:00
return strncmp(c1, c2, len);
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
int APP_CC
g_strcasecmp(const char* c1, const char* c2)
{
#if defined(_WIN32)
return stricmp(c1, c2);
#else
return strcasecmp(c1, c2);
#endif
}
2005-04-03 00:54:54 +04:00
/*****************************************************************************/
int APP_CC
g_strncasecmp(const char* c1, const char* c2, int len)
2005-04-03 00:54:54 +04:00
{
2005-11-07 03:04:11 +03:00
#if defined(_WIN32)
return strnicmp(c1, c2, len);
#else
2005-09-25 08:08:26 +04:00
return strncasecmp(c1, c2, len);
2005-11-07 03:04:11 +03:00
#endif
2005-04-03 00:54:54 +04:00
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_atoi(char* str)
{
return atoi(str);
}
2005-08-14 06:23:20 +04:00
/*****************************************************************************/
int APP_CC
g_pos(char* str, const char* to_find)
2005-08-14 06:23:20 +04:00
{
char* pp;
pp = strstr(str, to_find);
if (pp == 0)
{
return -1;
}
return (pp - str);
}
2005-01-07 03:56:38 +03:00
/*****************************************************************************/
long APP_CC
2005-06-28 07:04:36 +04:00
g_load_library(char* in)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-07-13 05:07:00 +04:00
return (long)LoadLibrary(in);
#else
2005-03-08 06:31:14 +03:00
return (long)dlopen(in, RTLD_LOCAL | RTLD_LAZY);
#endif
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_free_library(long lib)
2005-01-07 03:56:38 +03:00
{
if (lib == 0)
{
2005-01-07 03:56:38 +03:00
return 0;
}
#if defined(_WIN32)
2005-07-13 05:07:00 +04:00
return FreeLibrary((HMODULE)lib);
#else
2005-01-07 03:56:38 +03:00
return dlclose((void*)lib);
#endif
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
/* returns NULL if not found */
void* APP_CC
g_get_proc_address(long lib, const char* name)
2005-01-07 03:56:38 +03:00
{
if (lib == 0)
{
2005-01-07 03:56:38 +03:00
return 0;
}
#if defined(_WIN32)
2005-07-13 05:07:00 +04:00
return GetProcAddress((HMODULE)lib, name);
#else
2005-01-07 03:56:38 +03:00
return dlsym((void*)lib, name);
#endif
2005-01-07 03:56:38 +03:00
}
/*****************************************************************************/
int APP_CC
2005-06-28 07:04:36 +04:00
g_system(char* aexec)
2005-01-07 03:56:38 +03:00
{
#if defined(_WIN32)
2005-02-20 09:06:26 +03:00
return 0;
#else
2005-01-07 03:56:38 +03:00
return system(aexec);
2005-02-20 09:06:26 +03:00
#endif
2005-01-07 03:56:38 +03:00
}
2005-02-08 06:45:30 +03:00
/*****************************************************************************/
char* APP_CC
g_get_strerror(void)
{
#if defined(_WIN32)
return 0;
#else
return strerror(errno);
#endif
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
int APP_CC
g_execvp(const char* p1, char* args[])
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
return 0;
2005-07-10 04:25:31 +04:00
#else
return execvp(p1, args);
#endif
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
int APP_CC
g_execlp3(const char* a1, const char* a2, const char* a3)
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
return 0;
#else
2005-12-09 02:38:50 +03:00
return execlp(a1, a2, a3, (void*)0);
2005-07-10 04:25:31 +04:00
#endif
}
2005-02-08 06:45:30 +03:00
/*****************************************************************************/
void APP_CC
2005-06-28 07:04:36 +04:00
g_signal(int sig_num, void (*func)(int))
2005-02-08 06:45:30 +03:00
{
#if defined(_WIN32)
2005-02-20 09:06:26 +03:00
#else
2005-02-08 06:45:30 +03:00
signal(sig_num, func);
2005-02-20 09:06:26 +03:00
#endif
2005-02-08 06:45:30 +03:00
}
2005-07-10 04:25:31 +04:00
2005-07-11 04:41:07 +04:00
/*****************************************************************************/
void APP_CC
2005-07-11 04:41:07 +04:00
g_signal_child_stop(void (*func)(int))
{
#if defined(_WIN32)
#else
signal(SIGCHLD, func);
#endif
}
2005-08-14 06:23:20 +04:00
/*****************************************************************************/
void APP_CC
2005-08-14 06:23:20 +04:00
g_unset_signals(void)
{
2005-08-18 04:14:34 +04:00
#if defined(_WIN32)
#else
2005-08-14 06:23:20 +04:00
sigset_t mask;
sigemptyset(&mask);
sigprocmask(SIG_SETMASK, &mask, NULL);
2005-08-18 04:14:34 +04:00
#endif
2005-08-14 06:23:20 +04:00
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_fork(void)
{
#if defined(_WIN32)
return 0;
#else
return fork();
#endif
}
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_setgid(int pid)
{
#if defined(_WIN32)
return 0;
#else
return setgid(pid);
#endif
}
/*****************************************************************************/
/* returns error, zero is success, non zero is error */
int APP_CC
g_initgroups(const char* user, int gid)
{
#if defined(_WIN32)
return 0;
#else
return initgroups(user ,gid);
#endif
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_setuid(int pid)
{
#if defined(_WIN32)
return 0;
#else
return setuid(pid);
#endif
}
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_waitchild(void)
{
#if defined(_WIN32)
return 0;
#else
int wstat;
return waitpid(0, &wstat, WNOHANG);
#endif
}
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_waitpid(int pid)
{
#if defined(_WIN32)
return 0;
#else
return waitpid(pid, 0, 0);
#endif
}
/*****************************************************************************/
void APP_CC
2005-07-10 04:25:31 +04:00
g_clearenv(void)
{
#if defined(_WIN32)
#else
2005-10-11 08:35:07 +04:00
environ = 0;
2005-07-10 04:25:31 +04:00
#endif
}
/*****************************************************************************/
int APP_CC
g_setenv(const char* name, const char* value, int rewrite)
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
return 0;
#else
return setenv(name, value, rewrite);
#endif
}
2005-09-25 08:08:26 +04:00
/*****************************************************************************/
char* APP_CC
g_getenv(const char* name)
2005-09-25 08:08:26 +04:00
{
#if defined(_WIN32)
return 0;
#else
return getenv(name);
#endif
}
2005-07-10 04:25:31 +04:00
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_exit(int exit_code)
{
_exit(exit_code);
return 0;
}
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_getpid(void)
{
#if defined(_WIN32)
return 0;
#else
return getpid();
#endif
}
/*****************************************************************************/
int APP_CC
2005-07-10 04:25:31 +04:00
g_sigterm(int pid)
{
#if defined(_WIN32)
return 0;
#else
return kill(pid, SIGTERM);
#endif
}
/*****************************************************************************/
/* returns 0 if ok */
int APP_CC
g_getuser_info(const char* username, int* gid, int* uid, char* shell,
char* dir, char* gecos)
2005-07-10 04:25:31 +04:00
{
#if defined(_WIN32)
return 1;
2005-07-10 04:25:31 +04:00
#else
struct passwd* pwd_1;
pwd_1 = getpwnam(username);
if (pwd_1 != 0)
{
if (gid != 0)
{
*gid = pwd_1->pw_gid;
}
if (uid != 0)
{
*uid = pwd_1->pw_uid;
}
if (dir != 0)
{
g_strcpy(dir, pwd_1->pw_dir);
}
if (shell != 0)
{
g_strcpy(shell, pwd_1->pw_shell);
}
if (gecos != 0)
{
g_strcpy(gecos, pwd_1->pw_gecos);
}
return 0;
}
return 1;
#endif
}
/*****************************************************************************/
/* returns 0 if ok */
int APP_CC
g_getgroup_info(const char* groupname, int* gid)
{
#if defined(_WIN32)
return 1;
#else
struct group* g;
g = getgrnam(groupname);
if (g != 0)
{
if (gid != 0)
{
*gid = g->gr_gid;
}
return 0;
2005-07-10 04:25:31 +04:00
}
return 1;
2005-07-10 04:25:31 +04:00
#endif
}
/*****************************************************************************/
/* returns error */
/* if zero is returned, then ok is set */
int APP_CC
g_check_user_in_group(const char* username, int gid, int* ok)
{
#if defined(_WIN32)
return 1;
#else
struct group* groups;
int i;
groups = getgrgid(gid);
if (groups == 0)
{
return 1;
}
*ok = 0;
i = 0;
while (0 != groups->gr_mem[i])
{
if (0 == g_strcmp(groups->gr_mem[i], username))
{
*ok = 1;
break;
}
i++;
}
2005-07-10 04:25:31 +04:00
return 0;
#endif
2005-07-10 04:25:31 +04:00
}
2005-12-10 05:26:26 +03:00
/*****************************************************************************/
/* returns the time since the Epoch (00:00:00 UTC, January 1, 1970),
measured in seconds. */
int APP_CC
2005-12-10 05:26:26 +03:00
g_time1(void)
{
#if defined(_WIN32)
2006-09-30 03:50:28 +04:00
return GetTickCount() / 1000;
2005-12-10 05:26:26 +03:00
#else
return time(0);
#endif
}