1
0
mirror of https://github.com/samba-team/samba.git synced 2024-12-23 17:34:34 +03:00
samba-mirror/source3/lib/util_file.c
Volker Lendecke 3a1685db0b lib: Remove "includes.h" from util_file.c
Signed-off-by: Volker Lendecke <vl@samba.org>
Reviewed-by: Michael Adam <obnox@samba.org>
2016-02-23 22:03:17 +01:00

220 lines
4.8 KiB
C

/*
* Unix SMB/CIFS implementation.
* SMB parameters and setup
* Copyright (C) Andrew Tridgell 1992-1998 Modified by Jeremy Allison 1995.
*
* This program is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License as published by the Free
* Software Foundation; either version 3 of the License, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along with
* this program; if not, see <http://www.gnu.org/licenses/>.
*/
#include "replace.h"
#include "lib/util_file.h"
#include "lib/util/debug.h"
#include "lib/util/samba_util.h"
#include "lib/util/sys_rw.h"
#include "lib/sys_popen.h"
#include "lib/async_req/async_sock.h"
#include "lib/util/tevent_unix.h"
struct file_pload_state {
struct tevent_context *ev;
size_t maxsize;
int fd;
uint8_t *buf;
};
static int file_pload_state_destructor(struct file_pload_state *s);
static void file_pload_readable(struct tevent_req *subreq);
struct tevent_req *file_pload_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
const char *syscmd, size_t maxsize)
{
struct tevent_req *req, *subreq;
struct file_pload_state *state;
req = tevent_req_create(mem_ctx, &state, struct file_pload_state);
if (req == NULL) {
return NULL;
}
state->ev = ev;
state->maxsize = maxsize;
state->fd = sys_popen(syscmd);
if (state->fd == -1) {
tevent_req_error(req, errno);
return tevent_req_post(req, ev);
}
talloc_set_destructor(state, file_pload_state_destructor);
subreq = wait_for_read_send(state, state->ev, state->fd, false);
if (tevent_req_nomem(subreq, req)) {
return tevent_req_post(req, ev);
}
tevent_req_set_callback(subreq, file_pload_readable, req);
return req;
}
static int file_pload_state_destructor(struct file_pload_state *s)
{
if (s->fd != -1) {
sys_pclose(s->fd);
s->fd = -1;
}
return 0;
}
static void file_pload_readable(struct tevent_req *subreq)
{
struct tevent_req *req = tevent_req_callback_data(
subreq, struct tevent_req);
struct file_pload_state *state = tevent_req_data(
req, struct file_pload_state);
uint8_t buf[1024];
uint8_t *tmp;
ssize_t nread;
size_t bufsize;
int err;
bool ok;
ok = wait_for_read_recv(subreq, &err);
TALLOC_FREE(subreq);
if (!ok) {
tevent_req_error(req, err);
return;
}
nread = sys_read(state->fd, buf, sizeof(buf));
if (nread == -1) {
tevent_req_error(req, errno);
return;
}
if (nread == 0) {
tevent_req_done(req);
return;
}
bufsize = talloc_get_size(state->buf);
if (((bufsize + nread) < bufsize) ||
((bufsize + nread + 1) < bufsize)) {
/* overflow */
tevent_req_error(req, EMSGSIZE);
return;
}
if ((state->maxsize != 0) && ((bufsize + nread) > state->maxsize)) {
tevent_req_error(req, EMSGSIZE);
return;
}
tmp = talloc_realloc(state, state->buf, uint8_t, bufsize + nread + 1);
if (tevent_req_nomem(tmp, req)) {
return;
}
state->buf = tmp;
memcpy(state->buf + bufsize, buf, nread);
state->buf[bufsize+nread] = '\0';
subreq = wait_for_read_send(state, state->ev, state->fd, false);
if (tevent_req_nomem(subreq, req)) {
return;
}
tevent_req_set_callback(subreq, file_pload_readable, req);
}
int file_pload_recv(struct tevent_req *req, TALLOC_CTX *mem_ctx,
uint8_t **buf)
{
struct file_pload_state *state = tevent_req_data(
req, struct file_pload_state);
int err;
if (tevent_req_is_unix_error(req, &err)) {
return err;
}
*buf = talloc_move(mem_ctx, &state->buf);
tevent_req_received(req);
return 0;
}
/**
Load from a pipe into memory.
**/
static char *file_pload(const char *syscmd, size_t *size)
{
int fd, n;
char *p;
char buf[1024];
size_t total;
fd = sys_popen(syscmd);
if (fd == -1) {
return NULL;
}
p = NULL;
total = 0;
while ((n = sys_read(fd, buf, sizeof(buf))) > 0) {
p = talloc_realloc(NULL, p, char, total + n + 1);
if (!p) {
DEBUG(0,("file_pload: failed to expand buffer!\n"));
close(fd);
return NULL;
}
memcpy(p+total, buf, n);
total += n;
}
if (p) {
p[total] = 0;
}
/* FIXME: Perhaps ought to check that the command completed
* successfully (returned 0); if not the data may be
* truncated. */
sys_pclose(fd);
if (size) {
*size = total;
}
return p;
}
/**
Load a pipe into memory and return an array of pointers to lines in the data
must be freed with TALLOC_FREE.
**/
char **file_lines_pload(TALLOC_CTX *mem_ctx, const char *syscmd,
int *numlines)
{
char *p;
size_t size;
p = file_pload(syscmd, &size);
if (!p) {
return NULL;
}
return file_lines_parse(p, size, numlines, mem_ctx);
}