mirror of
https://github.com/samba-team/samba.git
synced 2025-01-25 06:04:04 +03:00
a2c1623827
The biggest thing is the integration of Lukes new nmbd. Its still largely untested, so we will really need some feedback I've also added auto prototype generation and cleaned up a lot of minor things as a result (This used to be commit 0d8dcfa13c527ec2c8aca39ba49c09e4e694b26c)
710 lines
17 KiB
C
710 lines
17 KiB
C
/*
|
|
Unix SMB/Netbios implementation.
|
|
Version 1.9.
|
|
NBT netbios routines and daemon - version 2
|
|
Copyright (C) Andrew Tridgell 1994-1995
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
Revision History:
|
|
|
|
14 jan 96: lkcl@pires.co.uk
|
|
added multiple workgroup domain master support
|
|
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "smb.h"
|
|
#include "loadparm.h"
|
|
#include "localnet.h"
|
|
|
|
extern int DEBUGLEVEL;
|
|
|
|
extern time_t StartupTime;
|
|
extern pstring myname;
|
|
extern pstring scope;
|
|
extern struct in_addr bcast_ip;
|
|
|
|
/* this is our browse master/backup cache database */
|
|
struct browse_cache_record *browserlist = NULL;
|
|
|
|
/* this is our domain/workgroup/server database */
|
|
struct domain_record *domainlist = NULL;
|
|
|
|
static BOOL updatedlists = True;
|
|
int updatecount=0;
|
|
|
|
int workgroup_count = 0; /* unique index key: one for each workgroup */
|
|
|
|
/* what server type are we currently */
|
|
|
|
#define DFLT_SERVER_TYPE (SV_TYPE_WORKSTATION | SV_TYPE_SERVER | \
|
|
SV_TYPE_TIME_SOURCE | SV_TYPE_SERVER_UNIX | \
|
|
SV_TYPE_PRINTQ_SERVER | SV_TYPE_POTENTIAL_BROWSER)
|
|
|
|
/* here are my election parameters */
|
|
|
|
/* NTAS uses 2, NT uses 1, WfWg uses 0 */
|
|
#define MAINTAIN_LIST 2
|
|
#define ELECTION_VERSION 2
|
|
|
|
#define MSBROWSE "\001\002__MSBROWSE__\002"
|
|
|
|
|
|
/****************************************************************************
|
|
add a workgroup into the domain list
|
|
**************************************************************************/
|
|
static void add_workgroup(struct work_record *work, struct domain_record *d)
|
|
{
|
|
struct work_record *w2;
|
|
|
|
if (!work || !d) return;
|
|
|
|
if (!d->workgrouplist)
|
|
{
|
|
d->workgrouplist = work;
|
|
work->prev = NULL;
|
|
work->next = NULL;
|
|
return;
|
|
}
|
|
|
|
for (w2 = d->workgrouplist; w2->next; w2 = w2->next);
|
|
|
|
w2->next = work;
|
|
work->next = NULL;
|
|
work->prev = w2;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
create a blank workgroup
|
|
**************************************************************************/
|
|
static struct work_record *make_workgroup(char *name)
|
|
{
|
|
struct work_record *work;
|
|
struct domain_record *d;
|
|
int t = -1;
|
|
|
|
if (!name || !name[0]) return NULL;
|
|
|
|
work = (struct work_record *)malloc(sizeof(*work));
|
|
if (!work) return(NULL);
|
|
|
|
StrnCpy(work->work_group,name,sizeof(work->work_group)-1);
|
|
work->serverlist = NULL;
|
|
|
|
work->ServerType = DFLT_SERVER_TYPE;
|
|
work->RunningElection = False;
|
|
work->ElectionCount = 0;
|
|
work->needelection = False;
|
|
work->needannounce = True;
|
|
|
|
/* make sure all token representations of workgroups are unique */
|
|
|
|
for (d = domainlist; d && t == -1; d = d->next)
|
|
{
|
|
struct work_record *w;
|
|
for (w = d->workgrouplist; w && t == -1; w = w->next)
|
|
{
|
|
if (strequal(w->work_group, work->work_group)) t = w->token;
|
|
}
|
|
}
|
|
|
|
if (t == -1)
|
|
{
|
|
work->token = ++workgroup_count;
|
|
}
|
|
else
|
|
{
|
|
work->token = t;
|
|
}
|
|
|
|
|
|
/* WfWg uses 01040b01 */
|
|
/* Win95 uses 01041501 */
|
|
/* NTAS uses ???????? */
|
|
work->ElectionCriterion = (MAINTAIN_LIST<<1)|(ELECTION_VERSION<<8);
|
|
work->ElectionCriterion |= (lp_os_level() << 24);
|
|
if (lp_domain_master())
|
|
{
|
|
work->ElectionCriterion |= 0x80;
|
|
}
|
|
|
|
return work;
|
|
}
|
|
|
|
|
|
/*******************************************************************
|
|
expire old servers in the serverlist
|
|
time of -1 indicates everybody dies
|
|
******************************************************************/
|
|
static void remove_old_servers(struct work_record *work, time_t t)
|
|
{
|
|
struct server_record *s;
|
|
struct server_record *nexts;
|
|
|
|
/* expire old entries in the serverlist */
|
|
for (s = work->serverlist; s; s = nexts)
|
|
{
|
|
if (t == -1 || (s->death_time && s->death_time < t))
|
|
{
|
|
DEBUG(3,("Removing dead server %s\n",s->serv.name));
|
|
updatedlists = True;
|
|
nexts = s->next;
|
|
|
|
if (s->prev) s->prev->next = s->next;
|
|
if (s->next) s->next->prev = s->prev;
|
|
|
|
if (work->serverlist == s) work->serverlist = s->next;
|
|
|
|
free(s);
|
|
}
|
|
else
|
|
{
|
|
nexts = s->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*******************************************************************
|
|
remove workgroups
|
|
******************************************************************/
|
|
struct work_record *remove_workgroup(struct domain_record *d, struct work_record *work)
|
|
{
|
|
struct work_record *ret_work = NULL;
|
|
|
|
if (!d || !work) return NULL;
|
|
|
|
DEBUG(3,("Removing old workgroup %s\n", work->work_group));
|
|
|
|
remove_old_servers(work, -1);
|
|
|
|
ret_work = work->next;
|
|
|
|
if (work->prev) work->prev->next = work->next;
|
|
if (work->next) work->next->prev = work->prev;
|
|
|
|
if (d->workgrouplist == work) d->workgrouplist = work->next;
|
|
|
|
free(work);
|
|
|
|
return ret_work;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
add a domain into the list
|
|
**************************************************************************/
|
|
static void add_domain(struct domain_record *d)
|
|
{
|
|
struct domain_record *d2;
|
|
|
|
if (!domainlist)
|
|
{
|
|
domainlist = d;
|
|
d->prev = NULL;
|
|
d->next = NULL;
|
|
return;
|
|
}
|
|
|
|
for (d2 = domainlist; d2->next; d2 = d2->next);
|
|
|
|
d2->next = d;
|
|
d->next = NULL;
|
|
d->prev = d2;
|
|
}
|
|
|
|
/***************************************************************************
|
|
add a browser into the list
|
|
**************************************************************************/
|
|
static void add_browse_cache(struct browse_cache_record *b)
|
|
{
|
|
struct browse_cache_record *b2;
|
|
|
|
if (!browserlist)
|
|
{
|
|
browserlist = b;
|
|
b->prev = NULL;
|
|
b->next = NULL;
|
|
return;
|
|
}
|
|
|
|
for (b2 = browserlist; b2->next; b2 = b2->next) ;
|
|
|
|
b2->next = b;
|
|
b->next = NULL;
|
|
b->prev = b2;
|
|
}
|
|
|
|
|
|
/***************************************************************************
|
|
add a server into the list
|
|
**************************************************************************/
|
|
static void add_server(struct work_record *work,struct server_record *s)
|
|
{
|
|
struct server_record *s2;
|
|
|
|
if (!work->serverlist) {
|
|
work->serverlist = s;
|
|
s->prev = NULL;
|
|
s->next = NULL;
|
|
return;
|
|
}
|
|
|
|
for (s2 = work->serverlist; s2->next; s2 = s2->next) ;
|
|
|
|
s2->next = s;
|
|
s->next = NULL;
|
|
s->prev = s2;
|
|
}
|
|
|
|
|
|
/*******************************************************************
|
|
remove old browse entries
|
|
******************************************************************/
|
|
void expire_browse_cache(time_t t)
|
|
{
|
|
struct browse_cache_record *b;
|
|
struct browse_cache_record *nextb;
|
|
|
|
/* expire old entries in the serverlist */
|
|
for (b = browserlist; b; b = nextb)
|
|
{
|
|
if (b->synced && b->sync_time < t)
|
|
{
|
|
DEBUG(3,("Removing dead cached browser %s\n",b->name));
|
|
nextb = b->next;
|
|
|
|
if (b->prev) b->prev->next = b->next;
|
|
if (b->next) b->next->prev = b->prev;
|
|
|
|
if (browserlist == b) browserlist = b->next;
|
|
|
|
free(b);
|
|
}
|
|
else
|
|
{
|
|
nextb = b->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
find a workgroup in the workgrouplist
|
|
only create it if the domain allows it, or the parameter 'add' insists
|
|
that it get created/added anyway. this allows us to force entries in
|
|
lmhosts file to be added.
|
|
**************************************************************************/
|
|
struct work_record *find_workgroupstruct(struct domain_record *d, fstring name, BOOL add)
|
|
{
|
|
struct work_record *ret, *work;
|
|
|
|
if (!d) return NULL;
|
|
|
|
DEBUG(4, ("workgroup search for %s: ", name));
|
|
|
|
if (strequal(name, "*"))
|
|
{
|
|
DEBUG(2,("add any workgroups: initiating browser search on %s\n",
|
|
inet_ntoa(d->bcast_ip)));
|
|
queue_netbios_pkt_wins(ClientNMB,NMB_QUERY, FIND_MASTER,
|
|
MSBROWSE,0x1,0,
|
|
True,False, d->bcast_ip);
|
|
return NULL;
|
|
}
|
|
|
|
for (ret = d->workgrouplist; ret; ret = ret->next)
|
|
{
|
|
if (!strcmp(ret->work_group,name))
|
|
{
|
|
DEBUG(4, ("found\n"));
|
|
return(ret);
|
|
}
|
|
}
|
|
|
|
DEBUG(4, ("not found: creating\n"));
|
|
|
|
if ((work = make_workgroup(name)))
|
|
{
|
|
if (lp_preferred_master() &&
|
|
strequal(lp_workgroup(), name) &&
|
|
ip_equal(d->bcast_ip, bcast_ip))
|
|
{
|
|
DEBUG(3, ("preferred master startup for %s\n", work->work_group));
|
|
work->needelection = True;
|
|
work->ElectionCriterion |= (1<<3);
|
|
}
|
|
if (!ip_equal(bcast_ip, d->bcast_ip))
|
|
{
|
|
work->needelection = False;
|
|
}
|
|
add_workgroup(work, d);
|
|
return(work);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
find a domain in the domainlist
|
|
**************************************************************************/
|
|
struct domain_record *find_domain(struct in_addr source_ip)
|
|
{
|
|
struct domain_record *d;
|
|
|
|
/* search through domain list for broadcast/netmask that matches
|
|
the source ip address */
|
|
|
|
for (d = domainlist; d; d = d->next)
|
|
{
|
|
if (same_net(source_ip, d->bcast_ip, d->mask_ip))
|
|
{
|
|
return(d);
|
|
}
|
|
}
|
|
|
|
return (NULL);
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
dump a copy of the workgroup/domain database
|
|
**************************************************************************/
|
|
static void dump_workgroups(void)
|
|
{
|
|
struct domain_record *d;
|
|
|
|
for (d = domainlist; d; d = d->next)
|
|
{
|
|
if (d->workgrouplist)
|
|
{
|
|
struct work_record *work;
|
|
|
|
DEBUG(3,("dump domain %15s: ", inet_ntoa(d->bcast_ip)));
|
|
DEBUG(3,(" %15s:\n", inet_ntoa(d->bcast_ip)));
|
|
|
|
for (work = d->workgrouplist; work; work = work->next)
|
|
{
|
|
if (work->serverlist)
|
|
{
|
|
struct server_record *s;
|
|
|
|
DEBUG(3,("\t%s(%d)\n", work->work_group, work->token));
|
|
for (s = work->serverlist; s; s = s->next)
|
|
{
|
|
DEBUG(3,("\t\t%s %8x (%s)\n",
|
|
s->serv.name, s->serv.type, s->serv.comment));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/****************************************************************************
|
|
create a domain entry
|
|
****************************************************************************/
|
|
static struct domain_record *make_domain(struct in_addr ip, struct in_addr mask)
|
|
{
|
|
struct domain_record *d;
|
|
d = (struct domain_record *)malloc(sizeof(*d));
|
|
|
|
if (!d) return(NULL);
|
|
|
|
bzero((char *)d,sizeof(*d));
|
|
|
|
DEBUG(4, ("making domain %s ", inet_ntoa(ip)));
|
|
DEBUG(4, ("%s\n", inet_ntoa(mask)));
|
|
|
|
d->bcast_ip = ip;
|
|
d->mask_ip = mask;
|
|
d->workgrouplist = NULL;
|
|
|
|
add_domain(d);
|
|
|
|
return d;
|
|
}
|
|
|
|
/****************************************************************************
|
|
add a domain entry. creates a workgroup, if necessary, and adds the domain
|
|
to the named a workgroup.
|
|
****************************************************************************/
|
|
struct domain_record *add_domain_entry(struct in_addr source_ip, struct in_addr source_mask,
|
|
char *name, BOOL add)
|
|
{
|
|
struct domain_record *d;
|
|
struct in_addr ip;
|
|
|
|
ip = *interpret_addr2("255.255.255.255");
|
|
|
|
if (zero_ip(source_ip)) source_ip = bcast_ip;
|
|
|
|
/* add the domain into our domain database */
|
|
if ((d = find_domain(source_ip)) ||
|
|
(d = make_domain(source_ip, source_mask)))
|
|
{
|
|
find_workgroupstruct(d, name, add);
|
|
|
|
/* add WORKGROUP(1e) and WORKGROUP(00) entries into name database
|
|
or register with WINS server, if it's our workgroup */
|
|
if (strequal(lp_workgroup(), name))
|
|
{
|
|
add_name_entry(name,0x1e,NB_ACTIVE|NB_GROUP);
|
|
add_name_entry(name,0x0 ,NB_ACTIVE|NB_GROUP);
|
|
}
|
|
|
|
DEBUG(3,("Added domain name entry %s at %s\n", name,inet_ntoa(ip)));
|
|
return d;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/****************************************************************************
|
|
add a browser entry
|
|
****************************************************************************/
|
|
struct browse_cache_record *add_browser_entry(char *name, int type, char *wg,
|
|
time_t ttl, struct in_addr ip)
|
|
{
|
|
BOOL newentry=False;
|
|
|
|
struct browse_cache_record *b;
|
|
|
|
/* search for the entry: if it's already in the cache, update that entry */
|
|
for (b = browserlist; b; b = b->next)
|
|
{
|
|
if (ip_equal(ip,b->ip) && strequal(b->group, wg)) break;
|
|
}
|
|
|
|
if (b && b->synced)
|
|
{
|
|
/* entries get left in the cache for a while. this stops sync'ing too
|
|
often if the network is large */
|
|
DEBUG(4, ("browser %s %s %s already sync'd at time %d\n",
|
|
b->name, b->group, inet_ntoa(b->ip), b->sync_time));
|
|
return NULL;
|
|
}
|
|
|
|
if (!b)
|
|
{
|
|
newentry = True;
|
|
b = (struct browse_cache_record *)malloc(sizeof(*b));
|
|
|
|
if (!b) return(NULL);
|
|
|
|
bzero((char *)b,sizeof(*b));
|
|
}
|
|
|
|
/* update the entry */
|
|
ttl = time(NULL)+ttl;
|
|
|
|
StrnCpy(b->name ,name,sizeof(b->name )-1);
|
|
StrnCpy(b->group,wg ,sizeof(b->group)-1);
|
|
strupper(b->name);
|
|
strupper(b->group);
|
|
|
|
b->ip = ip;
|
|
b->type = type;
|
|
|
|
if (newentry || ttl < b->sync_time) b->sync_time = ttl;
|
|
|
|
if (newentry)
|
|
{
|
|
b->synced = False;
|
|
add_browse_cache(b);
|
|
|
|
DEBUG(3,("Added cache entry %s %s(%2x) %s ttl %d\n",
|
|
wg, name, type, inet_ntoa(ip),ttl));
|
|
}
|
|
else
|
|
{
|
|
DEBUG(3,("Updated cache entry %s %s(%2x) %s ttl %d\n",
|
|
wg, name, type, inet_ntoa(ip),ttl));
|
|
}
|
|
|
|
return(b);
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
add a server entry
|
|
****************************************************************************/
|
|
struct server_record *add_server_entry(struct domain_record *d, struct work_record *work,
|
|
char *name,int servertype, int ttl,char *comment,
|
|
BOOL replace)
|
|
{
|
|
BOOL newentry=False;
|
|
struct server_record *s;
|
|
|
|
if (name[0] == '*')
|
|
{
|
|
return (NULL);
|
|
}
|
|
|
|
for (s = work->serverlist; s; s = s->next)
|
|
{
|
|
if (strequal(name,s->serv.name)) break;
|
|
}
|
|
|
|
if (s && !replace)
|
|
{
|
|
DEBUG(4,("Not replacing %s\n",name));
|
|
return(s);
|
|
}
|
|
|
|
updatedlists=True;
|
|
|
|
if (!s)
|
|
{
|
|
newentry = True;
|
|
s = (struct server_record *)malloc(sizeof(*s));
|
|
|
|
if (!s) return(NULL);
|
|
|
|
bzero((char *)s,sizeof(*s));
|
|
}
|
|
|
|
if (ip_equal(bcast_ip, d->bcast_ip) &&
|
|
strequal(lp_workgroup(),work->work_group))
|
|
{
|
|
servertype |= SV_TYPE_LOCAL_LIST_ONLY;
|
|
}
|
|
else
|
|
{
|
|
servertype &= ~SV_TYPE_LOCAL_LIST_ONLY;
|
|
}
|
|
|
|
/* update the entry */
|
|
StrnCpy(s->serv.name,name,sizeof(s->serv.name)-1);
|
|
StrnCpy(s->serv.comment,comment,sizeof(s->serv.comment)-1);
|
|
strupper(s->serv.name);
|
|
s->serv.type = servertype;
|
|
s->death_time = ttl?time(NULL)+ttl*3:0;
|
|
|
|
/* for a domain entry, the comment field refers to the server name */
|
|
|
|
if (s->serv.type & SV_TYPE_DOMAIN_ENUM) strupper(s->serv.comment);
|
|
|
|
if (newentry)
|
|
{
|
|
add_server(work, s);
|
|
|
|
DEBUG(3,("Added "));
|
|
}
|
|
else
|
|
{
|
|
DEBUG(3,("Updated "));
|
|
}
|
|
|
|
DEBUG(3,("server entry %s of type %x (%s) to %s %s\n",
|
|
name,servertype,comment,
|
|
work->work_group,inet_ntoa(d->bcast_ip)));
|
|
|
|
return(s);
|
|
}
|
|
|
|
|
|
/*******************************************************************
|
|
write out browse.dat
|
|
******************************************************************/
|
|
void write_browse_list(void)
|
|
{
|
|
struct domain_record *d;
|
|
|
|
pstring fname,fnamenew;
|
|
FILE *f;
|
|
|
|
if (!updatedlists) return;
|
|
|
|
dump_names();
|
|
dump_workgroups();
|
|
|
|
updatedlists = False;
|
|
updatecount++;
|
|
|
|
strcpy(fname,lp_lockdir());
|
|
trim_string(fname,NULL,"/");
|
|
strcat(fname,"/");
|
|
strcat(fname,SERVER_LIST);
|
|
strcpy(fnamenew,fname);
|
|
strcat(fnamenew,".");
|
|
|
|
f = fopen(fnamenew,"w");
|
|
|
|
if (!f)
|
|
{
|
|
DEBUG(4,("Can't open %s - %s\n",fnamenew,strerror(errno)));
|
|
return;
|
|
}
|
|
|
|
for (d = domainlist; d ; d = d->next)
|
|
{
|
|
struct work_record *work;
|
|
for (work = d->workgrouplist; work ; work = work->next)
|
|
{
|
|
struct server_record *s;
|
|
for (s = work->serverlist; s ; s = s->next)
|
|
{
|
|
fstring tmp;
|
|
|
|
/* don't list domains I don't have a master for */
|
|
if ((s->serv.type & SV_TYPE_DOMAIN_ENUM) &&
|
|
!s->serv.comment[0])
|
|
{
|
|
continue;
|
|
}
|
|
|
|
/* output server details, plus what workgroup/domain
|
|
they're in. without the domain information, the
|
|
combined list of all servers in all workgroups gets
|
|
sent to anyone asking about any workgroup! */
|
|
|
|
sprintf(tmp, "\"%s\"", s->serv.name);
|
|
fprintf(f, "%-25s ", tmp);
|
|
fprintf(f, "%08x ", s->serv.type);
|
|
sprintf(tmp, "\"%s\"", s->serv.comment);
|
|
fprintf(f, "%-30s", tmp);
|
|
fprintf(f, "\"%s\"\n", work->work_group);
|
|
}
|
|
}
|
|
}
|
|
|
|
fclose(f);
|
|
unlink(fname);
|
|
chmod(fnamenew,0644);
|
|
rename(fnamenew,fname);
|
|
DEBUG(3,("Wrote browse list %s\n",fname));
|
|
}
|
|
|
|
|
|
/*******************************************************************
|
|
expire old servers in the serverlist
|
|
******************************************************************/
|
|
void expire_servers(time_t t)
|
|
{
|
|
struct domain_record *d;
|
|
|
|
for (d = domainlist ; d ; d = d->next)
|
|
{
|
|
struct work_record *work;
|
|
|
|
for (work = d->workgrouplist; work; work = work->next)
|
|
{
|
|
remove_old_servers(work, t);
|
|
}
|
|
}
|
|
}
|
|
|