bug: when checking whether there is a PDC already on the current local
[Samba/gbeck.git] / source / namedbwork.c
blob0cfc47c41a8a80d92b9be114a004bc5290931aab
1 /*
2 Unix SMB/Netbios implementation.
3 Version 1.9.
4 NBT netbios routines and daemon - version 2
5 Copyright (C) Andrew Tridgell 1994-1996
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, write to the Free Software
19 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21 Revision History:
23 14 jan 96: lkcl@pires.co.uk
24 added multiple workgroup domain master support
26 04 jul 96: lkcl@pires.co.uk
27 created module namedbwork containing workgroup database functions
31 #include "includes.h"
32 #include "smb.h"
34 extern int ClientNMB;
36 extern int DEBUGLEVEL;
38 /* this is our domain/workgroup/server database */
39 extern struct subnet_record *subnetlist;
41 extern struct in_addr ipgrp;
43 int workgroup_count = 0; /* unique index key: one for each workgroup */
47 /****************************************************************************
48 add a workgroup into the domain list
49 **************************************************************************/
50 static void add_workgroup(struct work_record *work, struct subnet_record *d)
52 struct work_record *w2;
54 if (!work || !d) return;
56 if (!d->workgrouplist)
58 d->workgrouplist = work;
59 work->prev = NULL;
60 work->next = NULL;
61 return;
64 for (w2 = d->workgrouplist; w2->next; w2 = w2->next);
66 w2->next = work;
67 work->next = NULL;
68 work->prev = w2;
72 /****************************************************************************
73 create a blank workgroup
74 **************************************************************************/
75 static struct work_record *make_workgroup(char *name)
77 struct work_record *work;
78 struct subnet_record *d;
79 int t = -1;
81 if (!name || !name[0]) return NULL;
83 work = (struct work_record *)malloc(sizeof(*work));
84 if (!work) return(NULL);
86 StrnCpy(work->work_group,name,sizeof(work->work_group)-1);
87 work->serverlist = NULL;
89 work->ServerType = DFLT_SERVER_TYPE | SV_TYPE_POTENTIAL_BROWSER;
90 work->RunningElection = False;
91 work->ElectionCount = 0;
92 work->needelection = False;
93 work->needannounce = True;
94 work->mst_state = MST_POTENTIAL;
95 work->dom_state = DOMAIN_NONE;
96 work->log_state = LOGON_NONE;
98 /* make sure all token representations of workgroups are unique */
100 for (d = subnetlist; d && t == -1; d = d->next)
102 struct work_record *w;
103 for (w = d->workgrouplist; w && t == -1; w = w->next)
105 if (strequal(w->work_group, work->work_group)) t = w->token;
109 if (t == -1)
111 work->token = ++workgroup_count;
113 else
115 work->token = t;
119 /* WfWg uses 01040b01 */
120 /* Win95 uses 01041501 */
121 /* NTAS uses ???????? */
122 work->ElectionCriterion = (MAINTAIN_LIST<<1)|(ELECTION_VERSION<<8);
123 work->ElectionCriterion |= (lp_os_level() << 24);
124 if (lp_domain_master()) {
125 work->ElectionCriterion |= 0x80;
128 return work;
132 /*******************************************************************
133 remove workgroups
134 ******************************************************************/
135 struct work_record *remove_workgroup(struct subnet_record *d,
136 struct work_record *work,
137 BOOL remove_all_servers)
139 struct work_record *ret_work = NULL;
141 if (!d || !work) return NULL;
143 DEBUG(3,("Removing old workgroup %s\n", work->work_group));
145 ret_work = work->next;
147 remove_old_servers(work, -1, remove_all_servers);
149 if (!work->serverlist)
151 if (work->prev) work->prev->next = work->next;
152 if (work->next) work->next->prev = work->prev;
154 if (d->workgrouplist == work) d->workgrouplist = work->next;
156 free(work);
159 return ret_work;
163 /****************************************************************************
164 find a workgroup in the workgrouplist
165 only create it if the domain allows it, or the parameter 'add' insists
166 that it get created/added anyway. this allows us to force entries in
167 lmhosts file to be added.
168 **************************************************************************/
169 struct work_record *find_workgroupstruct(struct subnet_record *d,
170 fstring name, BOOL add)
172 struct work_record *ret, *work;
174 if (!d) return NULL;
176 DEBUG(4, ("workgroup search for %s: ", name));
178 if (strequal(name, "*"))
180 DEBUG(2,("add any workgroups: initiating browser search on %s\n",
181 inet_ntoa(d->bcast_ip)));
182 queue_netbios_pkt_wins(d,ClientNMB,NMB_QUERY, NAME_QUERY_FIND_MST,
183 MSBROWSE,0x1,0,0,0,NULL,NULL,
184 True,False, d->bcast_ip, d->bcast_ip);
185 return NULL;
188 for (ret = d->workgrouplist; ret; ret = ret->next) {
189 if (!strcmp(ret->work_group,name)) {
190 DEBUG(4, ("found\n"));
191 return(ret);
195 if (!add) {
196 DEBUG(4, ("not found\n"));
197 return NULL;
200 DEBUG(4,("not found: creating\n"));
202 if ((work = make_workgroup(name)))
204 if (!ip_equal(d->bcast_ip, ipgrp) &&
205 lp_preferred_master() &&
206 strequal(lp_workgroup(), name))
208 DEBUG(3, ("preferred master startup for %s\n", work->work_group));
209 work->needelection = True;
210 work->ElectionCriterion |= (1<<3);
212 add_workgroup(work, d);
213 return(work);
215 return NULL;
219 /****************************************************************************
220 dump a copy of the workgroup/domain database
221 **************************************************************************/
222 void dump_workgroups(void)
224 struct subnet_record *d;
226 for (d = subnetlist; d; d = d->next)
228 if (d->workgrouplist)
230 struct work_record *work;
232 DEBUG(4,("dump domain bcast=%15s: ", inet_ntoa(d->bcast_ip)));
233 DEBUG(4,(" netmask=%15s:\n", inet_ntoa(d->mask_ip)));
235 for (work = d->workgrouplist; work; work = work->next)
237 DEBUG(4,("\t%s(%d)\n", work->work_group, work->token));
238 if (work->serverlist)
240 struct server_record *s;
241 for (s = work->serverlist; s; s = s->next)
243 DEBUG(4,("\t\t%s %8x (%s)\n",
244 s->serv.name, s->serv.type, s->serv.comment));