2 * Unix SMB/CIFS implementation.
3 * SMB parameters and setup
4 * Copyright (C) Andrew Tridgell 1992-1998 Modified by Jeremy Allison 1995.
6 * This program is free software; you can redistribute it and/or modify it under
7 * the terms of the GNU General Public License as published by the Free
8 * Software Foundation; either version 3 of the License, or (at your option)
11 * This program is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
16 * You should have received a copy of the GNU General Public License along with
17 * this program; if not, see <http://www.gnu.org/licenses/>.
21 #include "lib/util/util_file.h"
22 #include "source3/lib/util_file.h"
23 #include "lib/util/debug.h"
24 #include "lib/util/samba_util.h"
25 #include "lib/util/sys_rw.h"
26 #include "lib/util/sys_popen.h"
27 #include "lib/async_req/async_sock.h"
28 #include "lib/util/tevent_unix.h"
30 struct file_ploadv_state
{
31 struct tevent_context
*ev
;
32 struct tevent_req
*subreq
;
38 static void file_ploadv_cleanup_fn(
39 struct tevent_req
*req
, enum tevent_req_state req_state
);
40 static void file_ploadv_readable(struct tevent_req
*subreq
);
42 struct tevent_req
*file_ploadv_send(TALLOC_CTX
*mem_ctx
,
43 struct tevent_context
*ev
,
44 char * const argl
[], size_t maxsize
)
46 struct tevent_req
*req
= NULL
;
47 struct file_ploadv_state
*state
= NULL
;
49 req
= tevent_req_create(mem_ctx
, &state
, struct file_ploadv_state
);
54 state
->maxsize
= maxsize
;
56 state
->fd
= sys_popenv(argl
);
57 if (state
->fd
== -1) {
58 tevent_req_error(req
, errno
);
59 return tevent_req_post(req
, ev
);
61 tevent_req_set_cleanup_fn(req
, file_ploadv_cleanup_fn
);
63 state
->subreq
= wait_for_read_send(state
, state
->ev
, state
->fd
, false);
64 if (tevent_req_nomem(state
->subreq
, req
)) {
65 return tevent_req_post(req
, ev
);
67 tevent_req_set_callback(state
->subreq
, file_ploadv_readable
, req
);
71 static void file_ploadv_cleanup_fn(
72 struct tevent_req
*req
, enum tevent_req_state req_state
)
74 struct file_ploadv_state
*state
= tevent_req_data(
75 req
, struct file_ploadv_state
);
77 TALLOC_FREE(state
->subreq
);
78 if (state
->fd
!= -1) {
79 sys_pclose(state
->fd
);
84 static void file_ploadv_readable(struct tevent_req
*subreq
)
86 struct tevent_req
*req
= tevent_req_callback_data(
87 subreq
, struct tevent_req
);
88 struct file_ploadv_state
*state
= tevent_req_data(
89 req
, struct file_ploadv_state
);
97 ok
= wait_for_read_recv(subreq
, &err
);
101 tevent_req_error(req
, err
);
105 nread
= sys_read(state
->fd
, buf
, sizeof(buf
));
107 tevent_req_error(req
, errno
);
111 tevent_req_done(req
);
115 bufsize
= talloc_get_size(state
->buf
);
118 * Last round we've added the trailing '\0'. Remove it
124 if (((bufsize
+ nread
) < bufsize
) ||
125 ((bufsize
+ nread
+ 1) < bufsize
)) {
127 tevent_req_error(req
, EMSGSIZE
);
131 if ((state
->maxsize
!= 0) && ((bufsize
+ nread
) > state
->maxsize
)) {
132 tevent_req_error(req
, EMSGSIZE
);
136 tmp
= talloc_realloc(state
, state
->buf
, uint8_t, bufsize
+ nread
+ 1);
137 if (tevent_req_nomem(tmp
, req
)) {
142 memcpy(state
->buf
+ bufsize
, buf
, nread
);
143 state
->buf
[bufsize
+nread
] = '\0';
145 state
->subreq
= wait_for_read_send(state
, state
->ev
, state
->fd
, false);
146 if (tevent_req_nomem(state
->subreq
, req
)) {
149 tevent_req_set_callback(state
->subreq
, file_ploadv_readable
, req
);
152 int file_ploadv_recv(struct tevent_req
*req
, TALLOC_CTX
*mem_ctx
,
155 struct file_ploadv_state
*state
= tevent_req_data(
156 req
, struct file_ploadv_state
);
159 if (tevent_req_is_unix_error(req
, &err
)) {
162 *buf
= talloc_move(mem_ctx
, &state
->buf
);
164 tevent_req_received(req
);
171 Load a pipe into memory and return an array of pointers to lines in the data
172 must be freed with TALLOC_FREE.
175 char **file_lines_ploadv(TALLOC_CTX
*mem_ctx
,
182 p
= file_ploadv(argl
, &size
);
187 return file_lines_parse(p
, size
, numlines
, mem_ctx
);