1 /* This file is part of the Project Athena Zephyr Notification System.
2 * It contains functions for the Client Manager subsystem of the Zephyr server.
4 * Created by: John T. Kohl
8 * Copyright (c) 1987,1988,1991 by the Massachusetts Institute of Technology.
9 * For copying and distribution information, see the file
13 #include <zephyr/mit-copyright.h>
15 #include <sys/socket.h>
17 #if !defined (lint) && !defined (SABER)
18 static const char rcsid_client_c[] =
25 * Code_t client_register(notice, who, client, server, wantdefaults)
27 * struct sockaddr_in *who;
28 * Client **client; (RETURN)
32 * Code_t client_deregister(client, host, flush)
37 * Client *client_find(who, unsigned int port)
38 * struct in_addr *host;
41 * void client_dump_clients(fp, clist)
47 * a client: allocate space, find or insert the address in the
48 * server's list of hosts, initialize and insert the client into
49 * the host's list of clients.
51 * This routine assumes that the client has not been registered yet.
52 * The caller should check by calling client_find.
56 static Client *client_bucket[HASHSIZE];
58 #define INET_HASH(host, port) ((htonl((host)->s_addr) + \
59 htons((unsigned short) (port))) % HASHSIZE)
62 client_register(ZNotice_t *notice,
69 /* chain the client's host onto this server's host list */
72 zdbug((LOG_DEBUG, "client_register: adding %s at %s/%d",
73 notice->z_sender, inet_ntoa(*host), ntohs(notice->z_port)));
77 return ZSRV_BADSUBPORT;
79 *client_p = client = client_find(host, notice->z_port);
81 *client_p = client = (Client *) malloc(sizeof(Client));
84 memset(&client->addr, 0, sizeof(struct sockaddr_in));
86 client->session_keyblock = NULL;
89 memset(&client->session_key, 0, sizeof(client->session_key));
92 client->last_send = 0;
93 client->last_ack = NOW;
94 client->addr.sin_family = AF_INET;
95 client->addr.sin_addr.s_addr = host->s_addr;
96 client->addr.sin_port = notice->z_port;
99 client->principal = make_string(notice->z_sender, 0);
100 Client_insert(&client_bucket[INET_HASH(&client->addr.sin_addr,
101 notice->z_port)], client);
104 /* Add default subscriptions only if this is not resulting from a brain
105 * dump, AND this request wants defaults. */
106 if (!bdumping && wantdefaults)
107 return subscr_def_subs(client);
113 * Deregister the client, freeing resources.
114 * Remove any packets in the nack queue, release subscriptions, release
115 * locations, and dequeue him from the host.
119 client_deregister(Client *client,
122 Client_delete(client);
123 nack_release(client);
124 subscr_cancel_client(client);
125 free_string(client->principal);
127 if (client->session_keyblock)
128 krb5_free_keyblock(Z_krb5_ctx, client->session_keyblock);
131 uloc_flush_client(&client->addr);
136 client_flush_host(struct in_addr *host)
139 Client *client, *next;
141 for (i = 0; i < HASHSIZE; i++) {
142 for (client = client_bucket[i]; client; client = next) {
144 if (client->addr.sin_addr.s_addr == host->s_addr)
145 client_deregister(client, 1);
152 client_send_clients(void)
158 for (i = 0; i < HASHSIZE; i++) {
159 /* Allow packets to be processed between rows of the hash table. */
160 if (packets_waiting()) {
162 bdump_concurrent = 1;
164 bdump_concurrent = 0;
167 for (client = client_bucket[i]; client; client = client->next) {
169 retval = subscr_send_subs(client);
170 if (retval != ZERR_NONE)
179 * dump info about clients in this clist onto the fp.
180 * assumed to be called with SIGFPE blocked
181 * (true if called from signal handler)
185 client_dump_clients(FILE *fp)
190 for (i = 0; i < HASHSIZE; i++) {
191 for (client = client_bucket[i]; client; client = client->next) {
192 fprintf(fp, "%s/%d (%s):\n", inet_ntoa(client->addr.sin_addr),
193 ntohs(client->addr.sin_port), client->principal->string);
194 subscr_dump_subs(fp, client->subs);
200 * find a client by host and port
204 client_find(struct in_addr *host,
210 hashval = INET_HASH(host, port);
211 for (client = client_bucket[hashval]; client; client = client->next) {
212 if (client->addr.sin_addr.s_addr == host->s_addr
213 && client->addr.sin_port == port)