mirror of
https://github.com/samba-team/samba.git
synced 2024-12-27 03:21:53 +03:00
52fa04d173
Jeremy.
(This used to be commit 313fdcd3e9
)
401 lines
11 KiB
C
401 lines
11 KiB
C
/*
|
|
Unix SMB/Netbios implementation.
|
|
Version 1.9.
|
|
NBT netbios routines and daemon - version 2
|
|
Copyright (C) Andrew Tridgell 1994-1997
|
|
|
|
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
|
|
|
|
04 jul 96: lkcl@pires.co.uk
|
|
created module namedbsubnet containing subnet database functions
|
|
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "smb.h"
|
|
|
|
extern int ClientNMB;
|
|
extern int ClientDGRAM;
|
|
extern int global_nmb_port;
|
|
|
|
extern int DEBUGLEVEL;
|
|
|
|
extern struct in_addr wins_ip;
|
|
extern struct in_addr ipzero;
|
|
|
|
extern pstring myname;
|
|
extern fstring myworkgroup;
|
|
extern char **my_netbios_names;
|
|
|
|
BOOL updatedlists = True;
|
|
int updatecount = 0;
|
|
|
|
/* local interfaces structure */
|
|
extern struct interface *local_interfaces;
|
|
|
|
/* this is our domain/workgroup/server database */
|
|
struct subnet_record *subnetlist = NULL;
|
|
|
|
/* WINS subnet - keep this separate so enumeration code doesn't
|
|
run onto it by mistake. */
|
|
struct subnet_record *wins_client_subnet = NULL;
|
|
|
|
extern uint16 nb_type; /* samba's NetBIOS name type */
|
|
|
|
/****************************************************************************
|
|
add a domain into the list
|
|
**************************************************************************/
|
|
static void add_subnet(struct subnet_record *d)
|
|
{
|
|
struct subnet_record *d2;
|
|
|
|
if (!subnetlist)
|
|
{
|
|
subnetlist = d;
|
|
d->prev = NULL;
|
|
d->next = NULL;
|
|
return;
|
|
}
|
|
|
|
for (d2 = subnetlist; d2->next; d2 = d2->next);
|
|
|
|
d2->next = d;
|
|
d->next = NULL;
|
|
d->prev = d2;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
find a subnet in the subnetlist that a given IP address could
|
|
match - not including WINS. Returns NULL if no match.
|
|
**************************************************************************/
|
|
struct subnet_record *find_subnet(struct in_addr ip)
|
|
{
|
|
struct subnet_record *d = NULL;
|
|
|
|
/* search through subnet list for broadcast/netmask that matches
|
|
the source ip address. */
|
|
|
|
for (d = FIRST_SUBNET; d; d = NEXT_SUBNET_EXCLUDING_WINS(d))
|
|
{
|
|
if (same_net(ip, d->bcast_ip, d->mask_ip))
|
|
break;
|
|
}
|
|
|
|
return d;
|
|
}
|
|
|
|
/****************************************************************************
|
|
find a subnet in the subnetlist - if the subnet is not found
|
|
then return the WINS client subnet.
|
|
**************************************************************************/
|
|
struct subnet_record *find_subnet_all(struct in_addr ip)
|
|
{
|
|
struct subnet_record *d = find_subnet(ip);
|
|
if(!d)
|
|
return wins_client_subnet;
|
|
return d;
|
|
}
|
|
|
|
/****************************************************************************
|
|
create a subnet entry
|
|
****************************************************************************/
|
|
static struct subnet_record *make_subnet(struct in_addr myip, struct in_addr bcast_ip,
|
|
struct in_addr mask_ip, BOOL add)
|
|
{
|
|
struct subnet_record *d = NULL;
|
|
int nmb_sock, dgram_sock;
|
|
|
|
/* Check if we are creating the WINS subnet - if so don't create
|
|
sockets, use the ClientNMB and ClientDGRAM sockets instead.
|
|
*/
|
|
|
|
if(ip_equal(bcast_ip, wins_ip))
|
|
{
|
|
nmb_sock = -1;
|
|
dgram_sock = -1;
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* Attempt to open the sockets on port 137/138 for this interface
|
|
* and bind them.
|
|
* Fail the subnet creation if this fails.
|
|
*/
|
|
|
|
if((nmb_sock = open_socket_in(SOCK_DGRAM, global_nmb_port,0, myip.s_addr)) == -1)
|
|
{
|
|
DEBUG(0,("make_subnet: Failed to open nmb socket on interface %s \
|
|
for port %d. Error was %s\n", inet_ntoa(myip), global_nmb_port, strerror(errno)));
|
|
return NULL;
|
|
}
|
|
|
|
if((dgram_sock = open_socket_in(SOCK_DGRAM,DGRAM_PORT,3, myip.s_addr)) == -1)
|
|
{
|
|
DEBUG(0,("make_subnet: Failed to open dgram socket on interface %s \
|
|
for port %d. Error was %s\n", inet_ntoa(myip), DGRAM_PORT, strerror(errno)));
|
|
return NULL;
|
|
}
|
|
|
|
/* Make sure we can broadcast from these sockets. */
|
|
set_socket_options(nmb_sock,"SO_BROADCAST");
|
|
set_socket_options(dgram_sock,"SO_BROADCAST");
|
|
|
|
}
|
|
|
|
d = (struct subnet_record *)malloc(sizeof(*d));
|
|
|
|
if (!d)
|
|
{
|
|
DEBUG(0,("make_subnet: malloc fail !\n"));
|
|
close(nmb_sock);
|
|
close(dgram_sock);
|
|
return(NULL);
|
|
}
|
|
|
|
bzero((char *)d,sizeof(*d));
|
|
|
|
DEBUG(4, ("making subnet %s ", inet_ntoa(bcast_ip)));
|
|
DEBUG(4, ("%s\n", inet_ntoa(mask_ip)));
|
|
|
|
d->bcast_ip = bcast_ip;
|
|
d->mask_ip = mask_ip;
|
|
d->myip = myip;
|
|
d->nmb_sock = nmb_sock;
|
|
d->dgram_sock = dgram_sock;
|
|
d->workgrouplist = NULL;
|
|
|
|
if(add)
|
|
add_subnet(d);
|
|
|
|
return d;
|
|
}
|
|
|
|
/****************************************************************************
|
|
add a domain entry. creates a workgroup, if necessary, and adds the domain
|
|
to the named a workgroup.
|
|
****************************************************************************/
|
|
static struct subnet_record *add_subnet_entry(struct in_addr myip,
|
|
struct in_addr bcast_ip,
|
|
struct in_addr mask_ip, char *name,
|
|
BOOL create_subnets, BOOL add)
|
|
{
|
|
struct subnet_record *d = NULL;
|
|
|
|
if (zero_ip(bcast_ip))
|
|
bcast_ip = *iface_bcast(bcast_ip);
|
|
|
|
/* Note that we should also add into the WINS subnet as add_subnet_entry
|
|
should be called to add NetBIOS names and server entries on all
|
|
interfaces, including the WINS interface
|
|
*/
|
|
|
|
if(create_subnets == True)
|
|
{
|
|
/* Create new subnets. */
|
|
if((d = make_subnet(myip, bcast_ip, mask_ip, add)) == NULL)
|
|
{
|
|
DEBUG(0,("add_subnet_entry: Unable to create subnet %s\n",
|
|
inet_ntoa(bcast_ip) ));
|
|
return NULL;
|
|
}
|
|
return d;
|
|
}
|
|
if(ip_equal(bcast_ip, wins_ip))
|
|
return wins_client_subnet;
|
|
return find_subnet(bcast_ip);
|
|
}
|
|
|
|
/****************************************************************************
|
|
Add a workgroup into a subnet, and if it's our primary workgroup,
|
|
add the required names to it.
|
|
**************************************************************************/
|
|
|
|
void add_workgroup_to_subnet( struct subnet_record *d, char *group)
|
|
{
|
|
struct work_record *w = NULL;
|
|
|
|
DEBUG(5,("add_workgroup_to_subnet: Adding workgroup %s to subnet %s\n",
|
|
group, inet_ntoa(d->bcast_ip)));
|
|
|
|
/* This next statement creates the workgroup struct if it doesn't
|
|
already exist.
|
|
*/
|
|
if((w = find_workgroupstruct(d, group, True)) == NULL)
|
|
{
|
|
DEBUG(0,("add_workgroup_to_subnet: Unable to add workgroup %s to subnet %s\n",
|
|
group, inet_ntoa(d->bcast_ip) ));
|
|
return;
|
|
}
|
|
|
|
/* add WORKGROUP(00) entries into name database
|
|
or register with WINS server, if it's our workgroup.
|
|
*/
|
|
if (strequal(myworkgroup, group))
|
|
{
|
|
int n;
|
|
|
|
add_my_name_entry(d,group,0x0 ,nb_type|NB_ACTIVE|NB_GROUP);
|
|
|
|
/* Register the WORKGROUP<0x1e> name. */
|
|
add_my_name_entry(d,group,0x1e,nb_type|NB_ACTIVE|NB_GROUP);
|
|
|
|
/* Add all our server names to the workgroup list. We remove any
|
|
browser or logon server flags from all but the primary name.
|
|
*/
|
|
for( n = 0; my_netbios_names[n]; n++)
|
|
{
|
|
char *name = my_netbios_names[n];
|
|
int stype = w->ServerType;
|
|
|
|
if(!strequal(myname, name))
|
|
stype &= ~(SV_TYPE_MASTER_BROWSER|SV_TYPE_POTENTIAL_BROWSER|
|
|
SV_TYPE_DOMAIN_MASTER|SV_TYPE_DOMAIN_MEMBER);
|
|
|
|
add_server_entry(d,w,name,stype|SV_TYPE_LOCAL_LIST_ONLY,0,
|
|
lp_serverstring(),True);
|
|
DEBUG(3,("add_workgroup_to_subnet: Added server name entry %s \
|
|
to subnet %s\n", name, inet_ntoa(d->bcast_ip)));
|
|
}
|
|
}
|
|
}
|
|
|
|
/****************************************************************************
|
|
create subnet / workgroup / server entries
|
|
|
|
- add or create the subnet lists
|
|
- add or create the workgroup entries in each subnet entry
|
|
- register appropriate NetBIOS names for the workgroup entries
|
|
|
|
**************************************************************************/
|
|
void add_my_subnets(char *group)
|
|
{
|
|
static BOOL create_subnets = True;
|
|
struct subnet_record *d = NULL;
|
|
struct interface *i = NULL;
|
|
|
|
if (*group == '*') return;
|
|
|
|
/* Create subnets from all the local interfaces and thread them onto
|
|
the linked list.
|
|
*/
|
|
for (i = local_interfaces; i; i = i->next)
|
|
{
|
|
add_subnet_entry(i->ip, i->bcast,i->nmask,group, create_subnets, True);
|
|
}
|
|
|
|
/* If we are using WINS, then we must add the workgroup to the WINS
|
|
subnet. This is used as a place to keep collated server lists.
|
|
*/
|
|
|
|
/* Create the WINS subnet if we are using WINS - but don't thread it
|
|
onto the linked subnet list.
|
|
*/
|
|
if (lp_wins_support() || lp_wins_server())
|
|
{
|
|
struct in_addr wins_nmask = ipzero;
|
|
wins_client_subnet = add_subnet_entry(ipzero, wins_ip, wins_nmask, group, create_subnets, False);
|
|
}
|
|
|
|
/* Ensure we only create the subnets once. */
|
|
create_subnets = False;
|
|
|
|
/* Now we have created all the subnets - we can add the names
|
|
that make us a client member in the workgroup.
|
|
*/
|
|
for (d = FIRST_SUBNET; d; d = NEXT_SUBNET_INCLUDING_WINS(d))
|
|
add_workgroup_to_subnet(d, group);
|
|
}
|
|
|
|
/*******************************************************************
|
|
write out browse.dat
|
|
******************************************************************/
|
|
void write_browse_list(time_t t)
|
|
{
|
|
struct subnet_record *d;
|
|
pstring fname,fnamenew;
|
|
FILE *f;
|
|
|
|
static time_t lasttime = 0;
|
|
|
|
if (!lasttime) lasttime = t;
|
|
if (!updatedlists || t - lasttime < 5) return;
|
|
|
|
lasttime = t;
|
|
updatedlists = False;
|
|
updatecount++;
|
|
|
|
dump_names();
|
|
dump_workgroups();
|
|
|
|
pstrcpy(fname,lp_lockdir());
|
|
trim_string(fname,NULL,"/");
|
|
strcat(fname,"/");
|
|
strcat(fname,SERVER_LIST);
|
|
pstrcpy(fnamenew,fname);
|
|
strcat(fnamenew,".");
|
|
|
|
f = fopen(fnamenew,"w");
|
|
|
|
if (!f)
|
|
{
|
|
DEBUG(4,("Can't open %s - %s\n",fnamenew,strerror(errno)));
|
|
return;
|
|
}
|
|
|
|
for (d = FIRST_SUBNET; d ; d = NEXT_SUBNET_INCLUDING_WINS(d))
|
|
{
|
|
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));
|
|
}
|
|
|