Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 : async implementation of WINBINDD_WINS_BYNAME
4 : Copyright (C) Volker Lendecke 2011
5 :
6 : This program is free software; you can redistribute it and/or modify
7 : it under the terms of the GNU General Public License as published by
8 : the Free Software Foundation; either version 3 of the License, or
9 : (at your option) any later version.
10 :
11 : This program is distributed in the hope that it will be useful,
12 : but WITHOUT ANY WARRANTY; without even the implied warranty of
13 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 : GNU General Public License for more details.
15 :
16 : You should have received a copy of the GNU General Public License
17 : along with this program. If not, see <http://www.gnu.org/licenses/>.
18 : */
19 :
20 : #include "includes.h"
21 : #include "winbindd.h"
22 : #include "libsmb/namequery.h"
23 : #include "librpc/gen_ndr/ndr_winbind_c.h"
24 : #include "libsmb/nmblib.h"
25 : #include "lib/util/string_wrappers.h"
26 :
27 : struct winbindd_wins_byname_state {
28 : struct tevent_context *ev;
29 : struct winbindd_request *request;
30 : struct sockaddr_storage *addrs;
31 : size_t num_addrs;
32 : };
33 :
34 : static void winbindd_wins_byname_wins_done(struct tevent_req *subreq);
35 : static void winbindd_wins_byname_bcast_done(struct tevent_req *subreq);
36 :
37 5 : struct tevent_req *winbindd_wins_byname_send(TALLOC_CTX *mem_ctx,
38 : struct tevent_context *ev,
39 : struct winbindd_cli_state *cli,
40 : struct winbindd_request *request)
41 : {
42 : struct tevent_req *req, *subreq;
43 : struct winbindd_wins_byname_state *state;
44 :
45 5 : req = tevent_req_create(mem_ctx, &state,
46 : struct winbindd_wins_byname_state);
47 5 : if (req == NULL) {
48 0 : return NULL;
49 : }
50 5 : state->ev = ev;
51 5 : state->request = request;
52 :
53 : /* Ensure null termination */
54 5 : request->data.winsreq[sizeof(request->data.winsreq)-1]='\0';
55 :
56 5 : D_NOTICE("[%s (%u)] Winbind external command WINS_BYNAME start.\n"
57 : "Resolving wins byname for '%s'.\n",
58 : cli->client_name,
59 : (unsigned int)cli->pid,
60 : request->data.winsreq);
61 :
62 5 : subreq = resolve_wins_send(state, ev, state->request->data.winsreq,
63 : 0x20);
64 5 : if (tevent_req_nomem(subreq, req)) {
65 0 : return tevent_req_post(req, ev);
66 : }
67 5 : tevent_req_set_callback(subreq, winbindd_wins_byname_wins_done, req);
68 5 : return req;
69 : }
70 :
71 5 : static void winbindd_wins_byname_wins_done(struct tevent_req *subreq)
72 : {
73 5 : struct tevent_req *req = tevent_req_callback_data(
74 : subreq, struct tevent_req);
75 5 : struct winbindd_wins_byname_state *state = tevent_req_data(
76 : req, struct winbindd_wins_byname_state);
77 : NTSTATUS status;
78 :
79 5 : status = resolve_wins_recv(subreq, talloc_tos(), &state->addrs,
80 : &state->num_addrs, NULL);
81 5 : TALLOC_FREE(subreq);
82 5 : if (NT_STATUS_IS_OK(status)) {
83 0 : tevent_req_done(req);
84 0 : return;
85 : }
86 5 : subreq = name_resolve_bcast_send(state, state->ev,
87 5 : state->request->data.winsreq, 0x20);
88 5 : if (tevent_req_nomem(subreq, req)) {
89 0 : return;
90 : }
91 5 : tevent_req_set_callback(subreq, winbindd_wins_byname_bcast_done, req);
92 : }
93 :
94 5 : static void winbindd_wins_byname_bcast_done(struct tevent_req *subreq)
95 : {
96 5 : struct tevent_req *req = tevent_req_callback_data(
97 : subreq, struct tevent_req);
98 5 : struct winbindd_wins_byname_state *state = tevent_req_data(
99 : req, struct winbindd_wins_byname_state);
100 : NTSTATUS status;
101 :
102 5 : status = name_resolve_bcast_recv(subreq, talloc_tos(), &state->addrs,
103 : &state->num_addrs);
104 5 : TALLOC_FREE(subreq);
105 5 : if (tevent_req_nterror(req, status)) {
106 0 : return;
107 : }
108 5 : tevent_req_done(req);
109 : }
110 :
111 5 : NTSTATUS winbindd_wins_byname_recv(struct tevent_req *req,
112 : struct winbindd_response *presp)
113 : {
114 5 : struct winbindd_wins_byname_state *state = tevent_req_data(
115 : req, struct winbindd_wins_byname_state);
116 : char *response;
117 : NTSTATUS status;
118 : size_t i;
119 :
120 5 : if (tevent_req_is_nterror(req, &status)) {
121 0 : return status;
122 : }
123 :
124 5 : response = talloc_strdup(talloc_tos(), "");
125 5 : if (response == NULL) {
126 0 : return NT_STATUS_NO_MEMORY;
127 : }
128 :
129 5 : D_NOTICE("Winbind external command WINS_BYNAME end.\n"
130 : "Received %zu address(es).\n",
131 : state->num_addrs);
132 10 : for (i=0; i<state->num_addrs; i++) {
133 : char addr[INET6_ADDRSTRLEN];
134 5 : print_sockaddr(addr, sizeof(addr), &state->addrs[i]);
135 5 : D_NOTICE("%zu: %s\n", i, addr);
136 5 : response = talloc_asprintf_append_buffer(
137 : response, "%s%s", addr,
138 5 : i < (state->num_addrs-1) ? " " : "");
139 5 : if (response == NULL) {
140 0 : return NT_STATUS_NO_MEMORY;
141 : }
142 : }
143 :
144 5 : response = talloc_asprintf_append_buffer(
145 5 : response, "\t%s\n", state->request->data.winsreq);
146 5 : if (response == NULL) {
147 0 : return NT_STATUS_NO_MEMORY;
148 : }
149 :
150 5 : if (talloc_get_size(response) > sizeof(presp->data.winsresp)) {
151 0 : TALLOC_FREE(response);
152 0 : return NT_STATUS_MARSHALL_OVERFLOW;
153 : }
154 5 : fstrcpy(presp->data.winsresp, response);
155 5 : TALLOC_FREE(response);
156 5 : return NT_STATUS_OK;
157 : }
|