s4:ldb add support for permissive modify control
[Samba/gebeck_regimport.git] / libcli / nbt / lmhosts.c
blob317ccc556c73e06424561a0ca10908ca468c3569
1 /*
2 Unix SMB/CIFS implementation.
4 manipulate nbt name structures
6 Copyright (C) Andrew Tridgell 1994-1998
7 Copyright (C) Jeremy Allison 2007
8 Copyright (C) Andrew Bartlett 2009.
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 3 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program. If not, see <http://www.gnu.org/licenses/>.
24 #include "includes.h"
25 #include "lib/util/xfile.h"
26 #include "lib/util/util_net.h"
27 #include "system/filesys.h"
28 #include "system/network.h"
30 /********************************************************
31 Start parsing the lmhosts file.
32 *********************************************************/
34 XFILE *startlmhosts(const char *fname)
36 XFILE *fp = x_fopen(fname,O_RDONLY, 0);
37 if (!fp) {
38 DEBUG(4,("startlmhosts: Can't open lmhosts file %s. "
39 "Error was %s\n",
40 fname, strerror(errno)));
41 return NULL;
43 return fp;
46 /********************************************************
47 Parse the next line in the lmhosts file.
48 *********************************************************/
50 bool getlmhostsent(TALLOC_CTX *ctx, XFILE *fp, char **pp_name, int *name_type,
51 struct sockaddr_storage *pss)
53 char line[1024];
55 *pp_name = NULL;
57 while(!x_feof(fp) && !x_ferror(fp)) {
58 char *ip = NULL;
59 char *flags = NULL;
60 char *extra = NULL;
61 char *name = NULL;
62 const char *ptr;
63 char *ptr1 = NULL;
64 int count = 0;
66 *name_type = -1;
68 if (!fgets_slash(line,sizeof(line),fp)) {
69 continue;
72 if (*line == '#') {
73 continue;
76 ptr = line;
78 if (next_token_talloc(ctx, &ptr, &ip, NULL))
79 ++count;
80 if (next_token_talloc(ctx, &ptr, &name, NULL))
81 ++count;
82 if (next_token_talloc(ctx, &ptr, &flags, NULL))
83 ++count;
84 if (next_token_talloc(ctx, &ptr, &extra, NULL))
85 ++count;
87 if (count <= 0)
88 continue;
90 if (count > 0 && count < 2) {
91 DEBUG(0,("getlmhostsent: Ill formed hosts line [%s]\n",
92 line));
93 continue;
96 if (count >= 4) {
97 DEBUG(0,("getlmhostsent: too many columns "
98 "in lmhosts file (obsolete syntax)\n"));
99 continue;
102 if (!flags) {
103 flags = talloc_strdup(ctx, "");
104 if (!flags) {
105 continue;
109 DEBUG(4, ("getlmhostsent: lmhost entry: %s %s %s\n",
110 ip, name, flags));
112 if (strchr_m(flags,'G') || strchr_m(flags,'S')) {
113 DEBUG(0,("getlmhostsent: group flag "
114 "in lmhosts ignored (obsolete)\n"));
115 continue;
118 if (!interpret_string_addr(pss, ip, AI_NUMERICHOST)) {
119 DEBUG(0,("getlmhostsent: invalid address "
120 "%s.\n", ip));
123 /* Extra feature. If the name ends in '#XX',
124 * where XX is a hex number, then only add that name type. */
125 if((ptr1 = strchr_m(name, '#')) != NULL) {
126 char *endptr;
127 ptr1++;
129 *name_type = (int)strtol(ptr1, &endptr, 16);
130 if(!*ptr1 || (endptr == ptr1)) {
131 DEBUG(0,("getlmhostsent: invalid name "
132 "%s containing '#'.\n", name));
133 continue;
136 *(--ptr1) = '\0'; /* Truncate at the '#' */
139 *pp_name = talloc_strdup(ctx, name);
140 if (!*pp_name) {
141 return false;
143 return true;
146 return false;
149 /********************************************************
150 Finish parsing the lmhosts file.
151 *********************************************************/
153 void endlmhosts(XFILE *fp)
155 x_fclose(fp);
158 /********************************************************
159 Resolve via "lmhosts" method.
160 *********************************************************/
162 NTSTATUS resolve_lmhosts_file_as_sockaddr(const char *lmhosts_file,
163 const char *name, int name_type,
164 TALLOC_CTX *mem_ctx,
165 struct sockaddr_storage **return_iplist,
166 int *return_count)
169 * "lmhosts" means parse the local lmhosts file.
172 XFILE *fp;
173 char *lmhost_name = NULL;
174 int name_type2;
175 struct sockaddr_storage return_ss;
176 NTSTATUS status = NT_STATUS_DOMAIN_CONTROLLER_NOT_FOUND;
177 TALLOC_CTX *ctx = NULL;
179 *return_iplist = NULL;
180 *return_count = 0;
182 DEBUG(3,("resolve_lmhosts: "
183 "Attempting lmhosts lookup for name %s<0x%x>\n",
184 name, name_type));
186 fp = startlmhosts(lmhosts_file);
188 if ( fp == NULL )
189 return NT_STATUS_NO_SUCH_FILE;
191 ctx = talloc_new(mem_ctx);
192 if (!ctx) {
193 endlmhosts(fp);
194 return NT_STATUS_NO_MEMORY;
197 while (getlmhostsent(ctx, fp, &lmhost_name, &name_type2, &return_ss)) {
199 if (!strequal(name, lmhost_name)) {
200 TALLOC_FREE(lmhost_name);
201 continue;
204 if ((name_type2 != -1) && (name_type != name_type2)) {
205 TALLOC_FREE(lmhost_name);
206 continue;
209 *return_iplist = talloc_realloc(ctx, (*return_iplist),
210 struct sockaddr_storage,
211 (*return_count)+1);
213 if ((*return_iplist) == NULL) {
214 TALLOC_FREE(ctx);
215 endlmhosts(fp);
216 DEBUG(3,("resolve_lmhosts: talloc_realloc fail !\n"));
217 return NT_STATUS_NO_MEMORY;
220 (*return_iplist)[*return_count] = return_ss;
221 *return_count += 1;
223 /* we found something */
224 status = NT_STATUS_OK;
226 /* Multiple names only for DC lookup */
227 if (name_type != 0x1c)
228 break;
231 talloc_steal(mem_ctx, *return_iplist);
232 TALLOC_FREE(ctx);
233 endlmhosts(fp);
234 return status;