clamd/dazukoio.c
6d6e8271
 #if HAVE_CONFIG_H
 #include "clamav-config.h"
 #endif
 
e3aaff8e
 #ifdef CLAMUKO
32fc1d7b
 /* Dazuko Interface. Interace with Dazuko for file access control.
    Written by John Ogness <jogness@antivir.de>
e3aaff8e
 
8d5e0ac0
    Copyright (c) 2002, 2003, 2004 H+BEDV Datentechnik GmbH
32fc1d7b
    All rights reserved.
 
    Redistribution and use in source and binary forms, with or without
    modification, are permitted provided that the following conditions
    are met:
 
    1. Redistributions of source code must retain the above copyright notice,
    this list of conditions and the following disclaimer.
 
    2. Redistributions in binary form must reproduce the above copyright notice,
    this list of conditions and the following disclaimer in the documentation
    and/or other materials provided with the distribution.
 
    3. Neither the name of Dazuko nor the names of its contributors may be used
    to endorse or promote products derived from this software without specific
    prior written permission.
 
    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
    AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
    IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
    ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
    LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
    CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
    SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
    INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
    CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
    ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
    POSSIBILITY OF SUCH DAMAGE.
 */
 
 #include <stdio.h>
e3aaff8e
 #include <stdlib.h>
 #include <sys/types.h>
 #include <fcntl.h>
 #include <string.h>
c7029064
 #ifdef HAVE_STRINGS_H
 #include <strings.h>
 #endif
e3aaff8e
 #include <unistd.h>
32fc1d7b
 #include "dazukoio_xp.h"
e3aaff8e
 #include "dazukoio.h"
 
32fc1d7b
 #if !defined(NO_COMPAT12)
 #include "dazukoio_compat12.h"
 #endif
 
 #define ITOA_SIZE	32
e3aaff8e
 
32fc1d7b
 dazuko_id_t	*_GLOBAL_DAZUKO = NULL;
 
 #if !defined(NO_COMPAT12)
 char		_GLOBAL_DAZUKO_COMPAT12 = 0;
 #endif
 
 static inline char char_to_hex(char c)
e3aaff8e
 {
32fc1d7b
 	/* ugly, but fast */
 
 	switch (c)
 	{
 		case '1': return 1;
 		case '2': return 2;
 		case '3': return 3;
 		case '4': return 4;
 		case '5': return 5;
 		case '6': return 6;
 		case '7': return 7;
 		case '8': return 8;
 		case '9': return 9;
 		case 'a': case 'A': return 10;
 		case 'b': case 'B': return 11;
 		case 'c': case 'C': return 12;
 		case 'd': case 'D': return 13;
 		case 'e': case 'E': return 14;
 		case 'f': case 'F': return 15;
 	}
 
 	return 0;
 }
 
 static void unescape_string(char *string)
 {
 	char	*p;
 
 	for (p=string ; *p ; p++)
 	{
 		/* check if we have \x */
 		if ((*p == '\\') && (*(p+1) == 'x'))
 		{
 			/* this is not cheap, but it should not occur often */
 
 			/* check if we have two more values following \x */
 			if (*(p+2) && *(p+3))
 			{
 				*p = char_to_hex(*(p+2));
 				*p <<= 4;
 				*p |= char_to_hex(*(p+3));
 
 				memmove(p + 1, p + 4, strlen(p+4) + 1);
 			}
 		}
 	}
 }
 
 static int get_value(const char *key, const char *string, char *buffer, size_t buffer_size)
 {
 	const char	*p1;
 	const char	*p2;
 	size_t		size;
 
 	if (buffer == NULL || buffer_size < 1)
 		return -1;
 
 	buffer[0] = 0;
e3aaff8e
 
32fc1d7b
 	if (key == NULL || string == NULL)
e3aaff8e
 		return -1;
 
32fc1d7b
 	p1 = strstr(string, key);
 	if (p1 == NULL)
 		return -1;
 
 	p1 += strlen(key);
 
 	for (p2=p1 ; *p2 && *p2!='\n' ; p2++)
 		continue;
 
 	size = p2 - p1;
 	if (size >= buffer_size)
 		size = buffer_size - 1;
 
 	memcpy(buffer, p1, size);
 
 	buffer[size] = 0;
 
 	return 0;
 }
 
 int dazukoRegister(const char *groupName, const char *mode)
 {
 	return dazukoRegister_TS(&_GLOBAL_DAZUKO, groupName, mode);
 }
 
 int dazukoRegister_TS(dazuko_id_t **dazuko_id, const char *groupName, const char *mode)
 {
 	struct dazuko_request	*request;
 	char			buffer[ITOA_SIZE];
 	char			regMode[3];
 	dazuko_id_t		*temp_id;
 	ssize_t			size;
 	int			write_mode = 0;
 #if !defined(NO_COMPAT12)
 	int			compat12_ret;
 #endif
 
 	if (dazuko_id == NULL)
 		return -1;
 
 	/* set default group name if one was not given */
 	if (groupName == NULL)
 		groupName = "_GENERIC";
 
 	/* set default mode if one was not given */
 	if (mode == NULL)
 		mode = "r";
 
 	if (strcasecmp(mode, "r") == 0)
 	{
658f19f8
 		strncpy(regMode, "R", sizeof(regMode));
72ce4b70
 		regMode[sizeof(regMode)-1]='\0';
32fc1d7b
 		write_mode = 0;
 	}
 	else if (strcasecmp(mode, "r+") == 0 || strcasecmp(mode, "rw") == 0)
 	{
658f19f8
 		strncpy(regMode, "RW", sizeof(regMode));
72ce4b70
 		regMode[sizeof(regMode)-1]='\0';
32fc1d7b
 		write_mode = 1;
 	}
 	else
e3aaff8e
 	{
 		return -1;
 	}
32fc1d7b
 	regMode[sizeof(regMode) - 1] = 0;
 
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 	{
 		compat12_ret = dazukoRegister_TS_compat12_wrapper(dazuko_id, groupName);
 
 		if (compat12_ret == 0)
 			(*dazuko_id)->write_mode = write_mode;
 
 		return compat12_ret;
 	}
 #endif
 
 	/* create temporary id */
 	temp_id = (dazuko_id_t *)malloc(sizeof(*temp_id));
 	if (temp_id == NULL)
 		return -1;
 
 	memset(temp_id, 0, sizeof(*temp_id));
 
 	/* open device */
 	temp_id->device = open("/dev/dazuko", O_RDWR);
 	if (temp_id->device < 0)
 	{
 		free(temp_id);
 		return -1;
 	}
 
 	/* read device major number */
 	memset(buffer, 0, sizeof(buffer));
 	if (read(temp_id->device, buffer, sizeof(buffer)-1) < 1)
 	{
 		close(temp_id->device);
 		free(temp_id);
 		return -1;
 	}
 
 	temp_id->dev_major = atoi(buffer);
 	if (temp_id->dev_major < 0)
 	{
 		close(temp_id->device);
 		free(temp_id);
 		return -1;
 	}
 
 	request = (struct dazuko_request *)malloc(sizeof(*request));
 	if (request == NULL)
 	{
 		close(temp_id->device);
 		free(temp_id);
 		return -1;
 	}
 
 	memset(request, 0, sizeof(*request));
 
 	request->type[0] = REGISTER;
 
 	size = 1 + 2 + 1 + strlen(regMode); /* \nRM=mode */
8d5e0ac0
 	size += 1 + 2 + 1 + strlen(groupName); /* \nGN=groupName */
32fc1d7b
 	size += 1; /* \0 */
 
 	request->buffer = (char *)malloc(size);
 	if (request->buffer == NULL)
 	{
 		close(temp_id->device);
 		free(temp_id);
 		free(request);
 		return -1;
 	}
 	snprintf(request->buffer, size, "\nRM=%s\nGN=%s", regMode, groupName);
 	request->buffer[size - 1] = 0;
 
 	request->buffer_size = strlen(request->buffer) + 1;
 
 	size = 4096;
 	request->reply_buffer = (char *)malloc(size);
 	if (request->reply_buffer == NULL)
 	{
 		close(temp_id->device);
 		free(temp_id);
 		free(request->buffer);
 		free(request);
 		return -1;
 	}
 	memset(request->reply_buffer, 0, size);
 	request->reply_buffer_size = size;
 
 	snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 	buffer[sizeof(buffer)-1] = 0;
 	size = strlen(buffer) + 1;
 
8d5e0ac0
 	if (write(temp_id->device, buffer, size) != size)
32fc1d7b
 	{
 		close(temp_id->device);
 		free(temp_id);
 		free(request->buffer);
 		free(request->reply_buffer);
 		free(request);
 
 #if !defined(NO_COMPAT12)
 		/* we try compat12 mode */
 		compat12_ret = dazukoRegister_TS_compat12_wrapper(dazuko_id, groupName);
 		if (compat12_ret == 0)
 		{
 			(*dazuko_id)->write_mode = write_mode;
 			_GLOBAL_DAZUKO_COMPAT12 = 1;
 		}
 
 		return compat12_ret;
 #else
 		return -1;
 #endif
 	}
 
 	if (get_value("\nID=", request->reply_buffer, buffer, sizeof(buffer)) != 0)
 	{
 		close(temp_id->device);
 		free(temp_id);
 		free(request->buffer);
 		free(request->reply_buffer);
 		free(request);
 
 #if !defined(NO_COMPAT12)
 		/* we try compat12 mode */
 		compat12_ret = dazukoRegister_TS_compat12_wrapper(dazuko_id, groupName);
 		if (compat12_ret == 0)
 		{
 			(*dazuko_id)->write_mode = write_mode;
 			_GLOBAL_DAZUKO_COMPAT12 = 1;
 		}
 
 		return compat12_ret;
 #else
 		return -1;
 #endif
 	}
 
 	temp_id->id = atoi(buffer);
 
 	if (temp_id->id < 0)
 	{
8d5e0ac0
 		close(temp_id->device);
 		free(temp_id);
32fc1d7b
 		free(request->buffer);
 		free(request->reply_buffer);
 		free(request);
 
 		return -1;
 	}
 
 	temp_id->write_mode = write_mode;
 
 	free(request->buffer);
 	free(request->reply_buffer);
 	free(request);
 
 	*dazuko_id = temp_id;
e3aaff8e
 
 	return 0;
 }
 
 int dazukoSetAccessMask(unsigned long accessMask)
 {
32fc1d7b
 	return dazukoSetAccessMask_TS(_GLOBAL_DAZUKO, accessMask);
 }
 
 int dazukoSetAccessMask_TS(dazuko_id_t *dazuko_id, unsigned long accessMask)
 {
 	struct dazuko_request	*request;
 	ssize_t			size;
 	char			buffer[ITOA_SIZE];
 
 	if (dazuko_id == NULL)
 		return -1;
 
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 		return dazukoSetAccessMask_TS_compat12(dazuko_id, accessMask);
 #endif
 
 	if (dazuko_id->device < 0 || dazuko_id->dev_major < 0 || dazuko_id->id < 0)
 		return -1;
 
 	request = (struct dazuko_request *)malloc(sizeof(*request));
 	if (request == NULL)
 		return -1;
 
 	memset(request, 0, sizeof(*request));
 
 	request->type[0] = SET_ACCESS_MASK;
 
 	size = 1 + 2 + 1 + ITOA_SIZE; /* \nID=id */
 	size += 1 + 2 + 1 + ITOA_SIZE; /* \nAM=accessMask */
 	size += 1; /* \0 */
 
 	request->buffer = (char *)malloc(size);
 	if (request->buffer == NULL)
 	{
 		free(request);
 		return -1;
 	}
 	snprintf(request->buffer, size, "\nID=%d\nAM=%lu", dazuko_id->id, accessMask);
 	request->buffer[size - 1] = 0;
e3aaff8e
 
32fc1d7b
 	request->buffer_size = strlen(request->buffer) + 1;
e3aaff8e
 
32fc1d7b
 	snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 	buffer[sizeof(buffer)-1] = 0;
 	size = strlen(buffer) + 1;
e3aaff8e
 
8d5e0ac0
 	if (write(dazuko_id->device, buffer, size) != size)
32fc1d7b
 	{
 		free(request->buffer);
 		free(request);
e3aaff8e
 		return -1;
32fc1d7b
 	}
 
 	free(request->buffer);
 	free(request);
e3aaff8e
 
 	return 0;
 }
 
32fc1d7b
 static int dazuko_set_path(dazuko_id_t *dazuko_id, const char *path, int type)
e3aaff8e
 {
32fc1d7b
 	struct dazuko_request	*request;
 	ssize_t			size;
 	char			buffer[ITOA_SIZE];
 
 	if (dazuko_id == NULL)
 		return -1;
 
 	if (dazuko_id->device < 0 || dazuko_id->dev_major < 0 || dazuko_id->id < 0)
 		return -1;
e3aaff8e
 
 	if (path == NULL)
 		return -1;
 
32fc1d7b
 	request = (struct dazuko_request *)malloc(sizeof(*request));
 	if (request == NULL)
 		return -1;
 
 	memset(request, 0, sizeof(*request));
 
 	request->type[0] = type;
e3aaff8e
 
32fc1d7b
 	size = 1 + 2 + 1 + ITOA_SIZE; /* \nID=id */
 	size += 1 + 2 + 1 + strlen(path); /* \nPT=path */
 	size += 1; /* \0 */
e3aaff8e
 
32fc1d7b
 	request->buffer = (char *)malloc(size);
 	if (request->buffer == NULL)
 	{
 		free(request);
e3aaff8e
 		return -1;
32fc1d7b
 	}
 	snprintf(request->buffer, size, "\nID=%d\nPT=%s", dazuko_id->id, path);
 	request->buffer[size - 1] = 0;
 
 	request->buffer_size = strlen(request->buffer) + 1;
 
 	snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 	buffer[sizeof(buffer)-1] = 0;
 	size = strlen(buffer) + 1;
 
8d5e0ac0
 	if (write(dazuko_id->device, buffer, size) != size)
32fc1d7b
 	{
 		free(request->buffer);
 		free(request);
 		return -1;
 	}
 
 	free(request->buffer);
 	free(request);
e3aaff8e
 
 	return 0;
 }
 
 int dazukoAddIncludePath(const char *path)
 {
32fc1d7b
 	return dazukoAddIncludePath_TS(_GLOBAL_DAZUKO, path);
 }
 
 int dazukoAddIncludePath_TS(dazuko_id_t *dazuko_id, const char *path)
 {
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 		return dazukoAddIncludePath_TS_compat12(dazuko_id, path);
 #endif
 
 	return dazuko_set_path(dazuko_id, path, ADD_INCLUDE_PATH);
e3aaff8e
 }
 
 int dazukoAddExcludePath(const char *path)
 {
32fc1d7b
 	return dazukoAddExcludePath_TS(_GLOBAL_DAZUKO, path);
 }
 
 int dazukoAddExcludePath_TS(dazuko_id_t *dazuko_id, const char *path)
 {
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 		return dazukoAddExcludePath_TS_compat12(dazuko_id, path);
 #endif
 
 	return dazuko_set_path(dazuko_id, path, ADD_EXCLUDE_PATH);
e3aaff8e
 }
 
 int dazukoRemoveAllPaths(void)
 {
32fc1d7b
 	return dazukoRemoveAllPaths_TS(_GLOBAL_DAZUKO);
 }
e3aaff8e
 
32fc1d7b
 int dazukoRemoveAllPaths_TS(dazuko_id_t *dazuko_id)
 {
 	struct dazuko_request	*request;
 	ssize_t			size;
 	char			buffer[ITOA_SIZE];
 
 	if (dazuko_id == NULL)
 		return -1;
 
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 		return dazukoRemoveAllPaths_TS_compat12(dazuko_id);
 #endif
 
 	if (dazuko_id->device < 0 || dazuko_id->dev_major < 0 || dazuko_id->id < 0)
 		return -1;
 
 	request = (struct dazuko_request *)malloc(sizeof(*request));
 	if (request == NULL)
 		return -1;
 
 	memset(request, 0, sizeof(*request));
 
 	request->type[0] = REMOVE_ALL_PATHS;
 
 	size = 1 + 2 + 1 + ITOA_SIZE; /* \nID=id */
 	size += 1; /* \0 */
 
 	request->buffer = (char *)malloc(size);
 	if (request->buffer == NULL)
 	{
 		free(request);
 		return -1;
 	}
 	snprintf(request->buffer, size, "\nID=%d", dazuko_id->id);
 	request->buffer[size - 1] = 0;
e3aaff8e
 
32fc1d7b
 	request->buffer_size = strlen(request->buffer) + 1;
e3aaff8e
 
32fc1d7b
 	snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 	buffer[sizeof(buffer)-1] = 0;
 	size = strlen(buffer) + 1;
 
8d5e0ac0
 	if (write(dazuko_id->device, buffer, size) != size)
32fc1d7b
 	{
 		free(request->buffer);
 		free(request);
e3aaff8e
 		return -1;
32fc1d7b
 	}
 
 	free(request->buffer);
 	free(request);
e3aaff8e
 
 	return 0;
32fc1d7b
 }
e3aaff8e
 
32fc1d7b
 int dazukoGetAccess(struct dazuko_access **acc)
 {
 	return dazukoGetAccess_TS(_GLOBAL_DAZUKO, acc);
e3aaff8e
 }
 
32fc1d7b
 int dazukoGetAccess_TS(dazuko_id_t *dazuko_id, struct dazuko_access **acc)
e3aaff8e
 {
32fc1d7b
 	struct dazuko_request	*request;
 	struct dazuko_access	*temp_acc;
 	ssize_t			size;
 	size_t			filename_size;
 	char			buffer[ITOA_SIZE];
 #if !defined(NO_COMPAT12)
 	int			compat12_ret;
 #endif
 
 	if (dazuko_id == NULL)
 		return -1;
 
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 	{
 		compat12_ret = dazukoGetAccess_TS_compat12_wrapper(dazuko_id, acc);
 
 		if (compat12_ret == 0 && !(dazuko_id->write_mode))
 		{
 			/* we are in read_only mode so we return the access immediately */
 
 			dazukoReturnAccess_TS_compat12_wrapper(dazuko_id, acc, 1, 0);
 
 			/* this could be dangerous, we do not check if the return was successfull! */
 		}
 
 		return compat12_ret;
 	}
 #endif
 
 	if (dazuko_id->device < 0 || dazuko_id->dev_major < 0 || dazuko_id->id < 0)
 		return -1;
 
e3aaff8e
 	if (acc == NULL)
 		return -1;
 
32fc1d7b
 	request = (struct dazuko_request *)malloc(sizeof(*request));
 	if (request == NULL)
 		return -1;
 
 	memset(request, 0, sizeof(*request));
 
 	request->type[0] = GET_AN_ACCESS;
 
 	size = 1 + 2 + 1 + ITOA_SIZE; /* \nID=id */
 	size += 1; /* \0 */
 
 	request->buffer = (char *)malloc(size);
 	if (request->buffer == NULL)
 	{
 		free(request);
 		return -1;
 	}
 	snprintf(request->buffer, size, "\nID=%d", dazuko_id->id);
 	request->buffer[size - 1] = 0;
 
 	request->buffer_size = strlen(request->buffer) + 1;
 
 	size = 1 + 2 + 1 + DAZUKO_FILENAME_MAX_LENGTH; /* \nFN=filename */
 	size += 1024; /* miscellaneous access attributes */
 	size += 1; /* \0 */
 	request->reply_buffer = (char *)malloc(size);
 	if (request->reply_buffer == NULL)
 	{
 		free(request->buffer);
 		free(request);
 		return -1;
 	}
 	memset(request->reply_buffer, 0, size);
 	request->reply_buffer_size = size;
 
 	temp_acc = (struct dazuko_access *)malloc(sizeof(*temp_acc));
 	if (temp_acc == NULL)
 	{
 		free(request->reply_buffer);
 		free(request->buffer);
 		free(request);
 		return -1;
 	}
 
 	memset(temp_acc, 0, sizeof(*temp_acc));
 
 	filename_size = DAZUKO_FILENAME_MAX_LENGTH + 1;
 	temp_acc->filename = (char *)malloc(filename_size);
 	if (temp_acc->filename == NULL)
 	{
 		free(temp_acc);
 		free(request->reply_buffer);
 		free(request->buffer);
 		free(request);
 		return -1;
 	}
 
 	snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 	buffer[sizeof(buffer)-1] = 0;
 	size = strlen(buffer) + 1;
e3aaff8e
 
8d5e0ac0
 	if (write(dazuko_id->device, buffer, size) != size)
32fc1d7b
 	{
 		free(temp_acc->filename);
 		free(temp_acc);
 		free(request->reply_buffer);
 		free(request->buffer);
 		free(request);
e3aaff8e
 		return -1;
32fc1d7b
 	}
 
 	if (request->reply_buffer_size_used > 0)
 	{
 		if (get_value("\nFN=", request->reply_buffer, temp_acc->filename, filename_size) == 0)
 		{
 			temp_acc->set_filename = 1;
 			unescape_string(temp_acc->filename);
 		}
 
 		if (get_value("\nEV=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->event = atoi(buffer);
 			temp_acc->set_event = 1;
 		}
 
 		if (get_value("\nFL=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->flags = atoi(buffer);
 			temp_acc->set_flags = 1;
 		}
 
 		if (get_value("\nMD=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->mode = atoi(buffer);
 			temp_acc->set_mode = 1;
 		}
 
 		if (get_value("\nUI=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->uid = atoi(buffer);
 			temp_acc->set_uid = 1;
 		}
 
 		if (get_value("\nPI=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->pid = atoi(buffer);
 			temp_acc->set_pid = 1;
 		}
 
 		if (get_value("\nFS=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->file_size = atol(buffer);
 			temp_acc->set_file_size = 1;
 		}
 
 		if (get_value("\nFU=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->file_uid = atoi(buffer);
 			temp_acc->set_file_uid = 1;
 		}
 
 		if (get_value("\nFG=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->file_gid = atoi(buffer);
 			temp_acc->set_file_gid = 1;
 		}
 
 		if (get_value("\nDT=", request->reply_buffer, buffer, sizeof(buffer)) == 0)
 		{
 			temp_acc->file_device = atoi(buffer);
 			temp_acc->set_file_device = 1;
 		}
 	}
 
 	free(request->reply_buffer);
 	free(request->buffer);
 	free(request);
 
 	*acc = temp_acc;
e3aaff8e
 
 	return 0;
 }
 
32fc1d7b
 int dazukoReturnAccess(struct dazuko_access **acc)
 {
 	return dazukoReturnAccess_TS(_GLOBAL_DAZUKO, acc);
 }
 
 int dazukoReturnAccess_TS(dazuko_id_t *dazuko_id, struct dazuko_access **acc)
e3aaff8e
 {
32fc1d7b
 	struct dazuko_request	*request;
 	ssize_t			size;
 	char			buffer[ITOA_SIZE];
 
 	if (dazuko_id == NULL)
 		return -1;
 
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 		return dazukoReturnAccess_TS_compat12_wrapper(dazuko_id, acc, dazuko_id->write_mode, 1);
 #endif
 
 	if (dazuko_id->device < 0 || dazuko_id->dev_major < 0 || dazuko_id->id < 0)
 		return -1;
 
e3aaff8e
 	if (acc == NULL)
 		return -1;
 
32fc1d7b
 	if (*acc == NULL)
e3aaff8e
 		return -1;
 
32fc1d7b
 	if (dazuko_id->write_mode)
 	{
 		request = (struct dazuko_request *)malloc(sizeof(*request));
 		if (request == NULL)
 			return -1;
 
 		memset(request, 0, sizeof(*request));
 
 		request->type[0] = RETURN_AN_ACCESS;
 
 		size = 1 + 2 + 1 + ITOA_SIZE; /* \nID=id */
 		size += 1 + 2 + 1 + ITOA_SIZE; /* \nDN=deny */
 		size += 1; /* \0 */
 
 		request->buffer = (char *)malloc(size);
 		if (request->buffer == NULL)
 		{
 			free(request);
 			return -1;
 		}
 		snprintf(request->buffer, size, "\nID=%d\nDN=%d", dazuko_id->id, (*acc)->deny == 0 ? 0 : 1);
 		request->buffer[size - 1] = 0;
 
 		request->buffer_size = strlen(request->buffer) + 1;
 
 		snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 		buffer[sizeof(buffer)-1] = 0;
 		size = strlen(buffer) + 1;
 
8d5e0ac0
 		if (write(dazuko_id->device, buffer, size) != size)
32fc1d7b
 		{
 			/* there could be big problems if this happens */
 
 			if ((*acc)->filename != NULL)
 				free((*acc)->filename);
 			free(*acc);
 			*acc = NULL;
 			free(request->buffer);
 			free(request);
 			return -1;
 		}
 
 		free(request->buffer);
 		free(request);
 	}
 
 	if ((*acc)->filename != NULL)
 		free((*acc)->filename);
 	free(*acc);
 	*acc = NULL;
 
e3aaff8e
 	return 0;
 }
 
 int dazukoUnregister(void)
 {
32fc1d7b
 	return dazukoUnregister_TS(&_GLOBAL_DAZUKO);
e3aaff8e
 }
 
32fc1d7b
 int dazukoUnregister_TS(dazuko_id_t **dazuko_id)
 {
 	struct dazuko_request	*request;
 	ssize_t			size;
 	char			buffer[ITOA_SIZE];
 
 	if (dazuko_id == NULL)
 		return -1;
 
 #if !defined(NO_COMPAT12)
 	if (_GLOBAL_DAZUKO_COMPAT12)
 		return dazukoUnregister_TS_compat12_wrapper(dazuko_id);
 #endif
 
 	if (*dazuko_id == NULL)
 		return -1;
 
 	if ((*dazuko_id)->device < 0)
 		return -1;
 
 	if ((*dazuko_id)->dev_major >= 0 && (*dazuko_id)->id >= 0)
 	{
 		request = (struct dazuko_request *)malloc(sizeof(*request));
 		if (request == NULL)
 			return -1;
 
 		memset(request, 0, sizeof(*request));
 
 		request->type[0] = UNREGISTER;
 
 		size = 1 + 2 + 1 + ITOA_SIZE; /* \nID=id */
 		size += 1; /* \0 */
 
 		request->buffer = (char *)malloc(size);
 		if (request->buffer == NULL)
 		{
 			free(request);
 			return -1;
 		}
 		snprintf(request->buffer, size, "\nID=%d", (*dazuko_id)->id);
 		request->buffer[size - 1] = 0;
 
 		request->buffer_size = strlen(request->buffer) + 1;
 
 		snprintf(buffer, sizeof(buffer), "\nRA=%lu", (unsigned long)request);
 		buffer[sizeof(buffer)-1] = 0;
 		size = strlen(buffer) + 1;
 
8d5e0ac0
 		if (write((*dazuko_id)->device, buffer, size) != size)
32fc1d7b
 		{
 			/* there could be big problems if this happens */
 
 			close((*dazuko_id)->device);
 			free(*dazuko_id);
 			*dazuko_id = NULL;
 			free(request->buffer);
 			free(request);
 			return -1;
 		}
 
 		free(request->buffer);
 		free(request);
 	}
 
 	close((*dazuko_id)->device);
 	free(*dazuko_id);
 	*dazuko_id = NULL;
 
 	return 0;
 }
e3aaff8e
 #endif