summaryrefslogtreecommitdiff
path: root/source3
diff options
context:
space:
mode:
Diffstat (limited to 'source3')
-rw-r--r--source3/include/proto.h6
-rw-r--r--source3/libsmb/clirap.c135
2 files changed, 92 insertions, 49 deletions
diff --git a/source3/include/proto.h b/source3/include/proto.h
index c802b69951..03b2277fd5 100644
--- a/source3/include/proto.h
+++ b/source3/include/proto.h
@@ -2695,6 +2695,12 @@ bool cli_qfileinfo(struct cli_state *cli, uint16_t fnum,
struct timespec *write_time,
struct timespec *change_time,
SMB_INO_T *ino);
+struct tevent_req *cli_qpathinfo_basic_send(TALLOC_CTX *mem_ctx,
+ struct event_context *ev,
+ struct cli_state *cli,
+ const char *fname);
+NTSTATUS cli_qpathinfo_basic_recv(struct tevent_req *req,
+ SMB_STRUCT_STAT *sbuf, uint32 *attributes);
NTSTATUS cli_qpathinfo_basic(struct cli_state *cli, const char *name,
SMB_STRUCT_STAT *sbuf, uint32 *attributes);
bool cli_qfileinfo_test(struct cli_state *cli, uint16_t fnum, int level, char **poutdata, uint32 *poutlen);
diff --git a/source3/libsmb/clirap.c b/source3/libsmb/clirap.c
index baa2dcd8e2..b1303f68d5 100644
--- a/source3/libsmb/clirap.c
+++ b/source3/libsmb/clirap.c
@@ -1144,68 +1144,105 @@ bool cli_qfileinfo(struct cli_state *cli, uint16_t fnum,
Send a qpathinfo BASIC_INFO call.
****************************************************************************/
-NTSTATUS cli_qpathinfo_basic(struct cli_state *cli, const char *name,
- SMB_STRUCT_STAT *sbuf, uint32 *attributes)
+struct cli_qpathinfo_basic_state {
+ uint32_t num_data;
+ uint8_t *data;
+};
+
+static void cli_qpathinfo_basic_done(struct tevent_req *subreq);
+
+struct tevent_req *cli_qpathinfo_basic_send(TALLOC_CTX *mem_ctx,
+ struct event_context *ev,
+ struct cli_state *cli,
+ const char *fname)
{
- unsigned int param_len = 0;
- uint32_t rdata_len;
- uint16_t setup[1];
- uint8_t *param, *rdata;
- uint8_t *p;
- char *path;
- int len;
- size_t nlen;
- TALLOC_CTX *frame = talloc_stackframe();
- NTSTATUS status;
+ struct tevent_req *req = NULL, *subreq = NULL;
+ struct cli_qpathinfo_basic_state *state = NULL;
- path = talloc_strdup(frame, name);
- if (!path) {
- TALLOC_FREE(frame);
- return NT_STATUS_NO_MEMORY;
+ req = tevent_req_create(mem_ctx, &state,
+ struct cli_qpathinfo_basic_state);
+ if (req == NULL) {
+ return NULL;
}
- /* cleanup */
-
- len = strlen(path);
- if ( path[len-1] == '\\' || path[len-1] == '/') {
- path[len-1] = '\0';
+ subreq = cli_qpathinfo_send(state, ev, cli, fname,
+ SMB_QUERY_FILE_BASIC_INFO,
+ 36, cli->max_xmit);
+ if (tevent_req_nomem(subreq, req)) {
+ return tevent_req_post(req, ev);
}
- nlen = 2*(strlen(path)+1);
+ tevent_req_set_callback(subreq, cli_qpathinfo_basic_done, req);
+ return req;
+}
- param = TALLOC_ARRAY(frame, uint8_t, 6+nlen+2);
- if (!param) {
- TALLOC_FREE(frame);
- return NT_STATUS_NO_MEMORY;
+static void cli_qpathinfo_basic_done(struct tevent_req *subreq)
+{
+ struct tevent_req *req = tevent_req_callback_data(
+ subreq, struct tevent_req);
+ struct cli_qpathinfo_basic_state *state = tevent_req_data(
+ req, struct cli_qpathinfo_basic_state);
+ NTSTATUS status;
+
+ status = cli_qpathinfo_recv(subreq, state, &state->data,
+ &state->num_data);
+ TALLOC_FREE(subreq);
+ if (!NT_STATUS_IS_OK(status)) {
+ tevent_req_nterror(req, status);
+ return;
}
- p = param;
- memset(param, '\0', 6);
+ tevent_req_done(req);
+}
- SSVAL(setup, 0, TRANSACT2_QPATHINFO);
- SSVAL(p, 0, SMB_QUERY_FILE_BASIC_INFO);
- p += 6;
- p += clistr_push(cli, p, path, nlen, STR_TERMINATE);
- param_len = PTR_DIFF(p, param);
+NTSTATUS cli_qpathinfo_basic_recv(struct tevent_req *req,
+ SMB_STRUCT_STAT *sbuf, uint32 *attributes)
+{
+ struct cli_qpathinfo_basic_state *state = tevent_req_data(
+ req, struct cli_qpathinfo_basic_state);
+ NTSTATUS status;
- status = cli_trans(talloc_tos(), cli, SMBtrans2, NULL, -1, 0, 0,
- setup, 1, 0,
- param, param_len, 2,
- NULL, 0, cli->max_xmit,
- NULL, 0, NULL,
- NULL, 0, NULL,
- &rdata, 36, &rdata_len);
- if (!NT_STATUS_IS_OK(status)) {
- TALLOC_FREE(frame);
+ if (tevent_req_is_nterror(req, &status)) {
return status;
}
- sbuf->st_ex_atime = interpret_long_date((char *)rdata+8);
- sbuf->st_ex_mtime = interpret_long_date((char *)rdata+16);
- sbuf->st_ex_ctime = interpret_long_date((char *)rdata+24);
-
- *attributes = IVAL( rdata, 32 );
+ sbuf->st_ex_atime = interpret_long_date((char *)state->data+8);
+ sbuf->st_ex_mtime = interpret_long_date((char *)state->data+16);
+ sbuf->st_ex_ctime = interpret_long_date((char *)state->data+24);
+ *attributes = IVAL(state->data, 32);
+ return NT_STATUS_OK;
+}
- TALLOC_FREE(rdata);
+NTSTATUS cli_qpathinfo_basic(struct cli_state *cli, const char *name,
+ SMB_STRUCT_STAT *sbuf, uint32 *attributes)
+{
+ TALLOC_CTX *frame = talloc_stackframe();
+ struct event_context *ev;
+ struct tevent_req *req;
+ NTSTATUS status = NT_STATUS_NO_MEMORY;
- return NT_STATUS_OK;
+ if (cli_has_async_calls(cli)) {
+ /*
+ * Can't use sync call while an async call is in flight
+ */
+ status = NT_STATUS_INVALID_PARAMETER;
+ goto fail;
+ }
+ ev = event_context_init(frame);
+ if (ev == NULL) {
+ goto fail;
+ }
+ req = cli_qpathinfo_basic_send(frame, ev, cli, name);
+ if (req == NULL) {
+ goto fail;
+ }
+ if (!tevent_req_poll_ntstatus(req, ev, &status)) {
+ goto fail;
+ }
+ status = cli_qpathinfo_basic_recv(req, sbuf, attributes);
+ fail:
+ TALLOC_FREE(frame);
+ if (!NT_STATUS_IS_OK(status)) {
+ cli_set_error(cli, status);
+ }
+ return status;
}
/****************************************************************************