struct dcerpc_binding_handle_raw_call_state {
const struct dcerpc_binding_handle_ops *ops;
- uint8_t *out_data;
- size_t out_length;
- uint32_t out_flags;
+ struct tevent_context *ev;
+ struct tevent_req *subreq;
};
static void dcerpc_binding_handle_raw_call_done(struct tevent_req *subreq);
{
struct tevent_req *req;
struct dcerpc_binding_handle_raw_call_state *state;
- struct tevent_req *subreq;
req = tevent_req_create(mem_ctx, &state,
struct dcerpc_binding_handle_raw_call_state);
return NULL;
}
state->ops = h->ops;
- state->out_data = NULL;
- state->out_length = 0;
- state->out_flags = 0;
+ state->ev = ev;
+
+ if (h->object != NULL) {
+ /*
+ * If an object is set on the binding handle,
+ * per request object passing is not allowed.
+ */
+ if (object != NULL) {
+ tevent_req_nterror(req, NT_STATUS_INVALID_HANDLE);
+ return tevent_req_post(req, ev);
+ }
- subreq = state->ops->raw_call_send(state, ev, h,
- object, opnum,
- in_flags, in_data, in_length);
- if (tevent_req_nomem(subreq, req)) {
+ /*
+ * We use the object from the binding handle
+ */
+ object = h->object;
+ }
+
+ state->subreq = state->ops->raw_call_send(state, ev, h,
+ object, opnum,
+ in_flags, in_data, in_length);
+ if (tevent_req_nomem(state->subreq, req)) {
return tevent_req_post(req, ev);
}
- tevent_req_set_callback(subreq, dcerpc_binding_handle_raw_call_done, req);
+ tevent_req_set_callback(state->subreq,
+ dcerpc_binding_handle_raw_call_done,
+ req);
return req;
}
static void dcerpc_binding_handle_raw_call_done(struct tevent_req *subreq)
{
- struct tevent_req *req = tevent_req_callback_data(subreq,
- struct tevent_req);
- struct dcerpc_binding_handle_raw_call_state *state =
- tevent_req_data(req,
- struct dcerpc_binding_handle_raw_call_state);
- NTSTATUS error;
+ struct tevent_req *req =
+ tevent_req_callback_data(subreq,
+ struct tevent_req);
- error = state->ops->raw_call_recv(subreq, state,
- &state->out_data,
- &state->out_length,
- &state->out_flags);
- TALLOC_FREE(subreq);
- if (tevent_req_nterror(req, error)) {
+ if (tevent_req_is_in_progress(subreq)) {
+ tevent_req_notify_callback(req);
return;
}
struct dcerpc_binding_handle_raw_call_state);
NTSTATUS error;
- if (tevent_req_is_nterror(req, &error)) {
+ if (!tevent_req_is_in_progress(req)) {
+ if (tevent_req_is_nterror(req, &error)) {
+ tevent_req_received(req);
+ return error;
+ }
+ }
+
+ error = state->ops->raw_call_recv(state->subreq,
+ mem_ctx,
+ out_data,
+ out_length,
+ out_flags);
+ if (!NT_STATUS_IS_OK(error)) {
tevent_req_received(req);
return error;
}
- *out_data = talloc_move(mem_ctx, &state->out_data);
- *out_length = state->out_length;
- *out_flags = state->out_flags;
+ if (tevent_req_is_in_progress(state->subreq)) {
+ return NT_STATUS_OK;
+ }
+
tevent_req_received(req);
return NT_STATUS_OK;
}
TALLOC_CTX *frame = talloc_stackframe();
struct tevent_context *ev;
struct tevent_req *subreq;
- NTSTATUS status;
+ NTSTATUS status = NT_STATUS_NO_MEMORY;
/*
* TODO: allow only one sync call
ev = samba_tevent_context_init(frame);
}
if (ev == NULL) {
- talloc_free(frame);
- return NT_STATUS_NO_MEMORY;
+ goto fail;
}
subreq = dcerpc_binding_handle_raw_call_send(frame, ev,
in_data,
in_length);
if (subreq == NULL) {
- talloc_free(frame);
- return NT_STATUS_NO_MEMORY;
+ goto fail;
}
if (!tevent_req_poll_ntstatus(subreq, ev, &status)) {
- talloc_free(frame);
- return status;
+ goto fail;
}
status = dcerpc_binding_handle_raw_call_recv(subreq,
out_data,
out_length,
out_flags);
- if (!NT_STATUS_IS_OK(status)) {
- talloc_free(frame);
- return status;
- }
-
+fail:
TALLOC_FREE(frame);
- return NT_STATUS_OK;
+ return status;
}
struct dcerpc_binding_handle_disconnect_state {
return NULL;
}
-#if 0 /* TODO: activate this when the callers are fixed */
if (table != h->table) {
tevent_req_nterror(req, NT_STATUS_INVALID_HANDLE);
return tevent_req_post(req, ev);
}
-#endif
if (opnum >= table->num_calls) {
tevent_req_nterror(req, NT_STATUS_INTERNAL_ERROR);