268 lines
6.7 KiB
C
268 lines
6.7 KiB
C
/* This file is part of the Project Athena Zephyr Notification System.
|
|
* It is one of the source files comprising zwgc, the Zephyr WindowGram
|
|
* client.
|
|
*
|
|
* Created by: Marc Horowitz <marc@athena.mit.edu>
|
|
*
|
|
* $Id$
|
|
*
|
|
* Copyright (c) 1989 by the Massachusetts Institute of Technology.
|
|
* For copying and distribution information, see the file
|
|
* "mit-copyright.h".
|
|
*/
|
|
|
|
#include <sysdep.h>
|
|
|
|
#if (!defined(lint) && !defined(SABER))
|
|
static const char rcsid_zephyr_c[] = "$Id$";
|
|
#endif
|
|
|
|
#include <zephyr/mit-copyright.h>
|
|
|
|
/****************************************************************************/
|
|
/* */
|
|
/* Module containing code dealing with zephyr: */
|
|
/* */
|
|
/****************************************************************************/
|
|
|
|
#include <zephyr/zephyr.h>
|
|
#include <sys/socket.h>
|
|
#include "new_string.h"
|
|
#include "zephyr.h"
|
|
#include "error.h"
|
|
#include "mux.h"
|
|
#include "subscriptions.h"
|
|
#include "variables.h"
|
|
#include "pointer.h"
|
|
#include "main.h"
|
|
#ifndef X_DISPLAY_MISSING
|
|
#include "X_driver.h"
|
|
#endif
|
|
#ifdef CMU_ZWGCPLUS
|
|
#include "plus.h"
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
extern int zwgc_debug;
|
|
#endif /* DEBUG */
|
|
|
|
static int zephyr_inited = 0;
|
|
static unsigned short zephyr_port = 0;
|
|
|
|
/*
|
|
* Internal Routine:
|
|
*
|
|
* string get_zwgc_port_number_filename()
|
|
* Effects: Returns the filename that the zwgc port # is/should be
|
|
* stored in, based on the user's uid & the environment
|
|
* variable WGFILE. The returned string points into a
|
|
* static buffer that may change on further calls to this
|
|
* routine or getenv. The returned string should not be
|
|
* modified in any way.
|
|
*/
|
|
|
|
static string
|
|
get_zwgc_port_number_filename(void)
|
|
{
|
|
static char buffer[40];
|
|
char *temp;
|
|
|
|
temp = getenv("WGFILE");
|
|
if (temp)
|
|
return(temp);
|
|
else {
|
|
sprintf(buffer, "/tmp/wg.%d", getuid());
|
|
return(buffer);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Write out the port number to the wg file.
|
|
*/
|
|
|
|
void
|
|
write_wgfile(void)
|
|
{
|
|
char *name = get_zwgc_port_number_filename();
|
|
FILE *port_file;
|
|
|
|
port_file = fopen(name, "w");
|
|
if (port_file) {
|
|
fprintf(port_file, "%d\n", zephyr_port);
|
|
fclose(port_file);
|
|
} else {
|
|
fprintf(stderr, "zwgc: error while opening %s for writing: ", name);
|
|
perror("");
|
|
}
|
|
}
|
|
|
|
/*
|
|
*
|
|
*/
|
|
|
|
struct notice_handler_ptr {
|
|
void (*notice_handler)(ZNotice_t *);
|
|
};
|
|
|
|
static void
|
|
handle_zephyr_input(struct notice_handler_ptr *nhp)
|
|
{
|
|
ZNotice_t *notice;
|
|
struct sockaddr_in from;
|
|
int complete_packets_ready;
|
|
|
|
for (;;) {
|
|
errno = 0;
|
|
if ( (complete_packets_ready=ZPending()) < 0 )
|
|
FATAL_TRAP( errno, "while calling ZPending()" );
|
|
|
|
if (complete_packets_ready==0)
|
|
return;
|
|
|
|
notice = malloc(sizeof(ZNotice_t));
|
|
|
|
TRAP( ZReceiveNotice(notice, &from), "while getting zephyr notice" );
|
|
if (!error_code) {
|
|
notice->z_auth = ZCheckAuthentication(notice, &from);
|
|
nhp->notice_handler(notice);
|
|
}
|
|
#ifdef CMU_ZWGCPLUS
|
|
if (get_list_refcount(notice) <= 0) {
|
|
/* no windows created */
|
|
if (!get_notice_fake(notice))
|
|
list_del_notice(notice);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/*
|
|
*
|
|
*/
|
|
|
|
void zephyr_init(void (*notice_handler)(ZNotice_t *))
|
|
{
|
|
struct notice_handler_ptr *nhp;
|
|
char *temp;
|
|
char *exposure;
|
|
char *tty = NULL;
|
|
FILE *port_file;
|
|
|
|
/*
|
|
* Initialize zephyr. If error, print error message & exit.
|
|
*/
|
|
nhp = malloc(sizeof(struct notice_handler_ptr));
|
|
if (!nhp) {
|
|
fprintf(stderr, "Out of memory setting up zephyr notice handler.\n");
|
|
exit(3);
|
|
}
|
|
FATAL_TRAP( ZInitialize(), "while initializing Zephyr" );
|
|
FATAL_TRAP( ZOpenPort(&zephyr_port), "while opening Zephyr port" );
|
|
|
|
/*
|
|
* Save away our port number in a special place so that zctl and
|
|
* other clients can send us control messages: <<<>>>
|
|
*/
|
|
temp = get_zwgc_port_number_filename();
|
|
port_file = fopen(temp, "r");
|
|
if (port_file) {
|
|
fprintf(stderr, "zwgc: windowgram file already exists. If you are\n");
|
|
fprintf(stderr, "zwgc: not already running zwgc, delete %s\n", temp);
|
|
fprintf(stderr, "zwgc: and try again.\n");
|
|
exit(1);
|
|
}
|
|
write_wgfile();
|
|
|
|
/* Set hostname and tty for locations. If we support X, use the
|
|
* display string for the default tty name. */
|
|
if (location_override)
|
|
tty = location_override;
|
|
#ifndef X_DISPLAY_MISSING
|
|
else if (x_dpy)
|
|
tty = DisplayString(x_dpy);
|
|
#endif
|
|
error_code = ZInitLocationInfo(NULL, tty);
|
|
TRAP( error_code, "while initializing location information" );
|
|
|
|
/*
|
|
* Retrieve the user's desired exposure level (from the zephyr variable
|
|
* "exposure"), convert it to the proper internal form then
|
|
* set the user's location using it. If the exposure level is
|
|
* not one of the allowed ones, print an error and treat it as
|
|
* EXPOSE_NONE.
|
|
*/
|
|
temp = ZGetVariable("exposure");
|
|
if (temp) {
|
|
if (!(exposure = ZParseExposureLevel(temp))) {
|
|
ERROR2("invalid exposure level %s, using exposure level none instead.\n", temp);
|
|
exposure = EXPOSE_NONE;
|
|
}
|
|
} else
|
|
exposure = EXPOSE_OPSTAFF;
|
|
error_code = ZSetLocation(exposure); /* <<<>>> */
|
|
if (error_code != ZERR_LOGINFAIL)
|
|
TRAP( error_code, "while setting location" );
|
|
|
|
/*
|
|
* If the exposure level isn't EXPOSE_NONE, turn on recieving notices.
|
|
* (this involves reading in the subscription file, etc.)
|
|
*/
|
|
if (string_Neq(exposure, EXPOSE_NONE))
|
|
zwgc_startup();
|
|
|
|
/*
|
|
* Set $realm to our realm and $user to our zephyr username:
|
|
*/
|
|
var_set_variable("realm", (char *)ZGetRealm()); /* XXX should propagate the
|
|
* const */
|
|
var_set_variable("user", ZGetSender());
|
|
|
|
/*
|
|
* <<<>>>
|
|
*/
|
|
nhp->notice_handler = notice_handler;
|
|
mux_add_input_source(ZGetFD(), (void (*)(void *))handle_zephyr_input, nhp);
|
|
zephyr_inited = 1;
|
|
return;
|
|
}
|
|
|
|
/*
|
|
*
|
|
*/
|
|
|
|
void finalize_zephyr(void) /* <<<>>> */
|
|
{
|
|
string temp;
|
|
|
|
if (zephyr_inited) {
|
|
/*
|
|
* Remove the file containing our port # since it is no longer needed:
|
|
*/
|
|
errno = 0;
|
|
temp = get_zwgc_port_number_filename();
|
|
unlink(temp);
|
|
if (errno) {
|
|
fprintf(stderr, "zwgc: error while trying to delete %s: ", temp);
|
|
perror("");
|
|
}
|
|
|
|
/*
|
|
* Cancel our subscriptions, unset our location, and close our zephyr
|
|
* connection:
|
|
*/
|
|
#ifdef DEBUG
|
|
if (zwgc_debug) {
|
|
TRAP( ZUnsetLocation(), "while unsetting location" );
|
|
TRAP( ZCancelSubscriptions(0), "while canceling subscriptions" );
|
|
} else {
|
|
#endif /* DEBUG */
|
|
(void) ZUnsetLocation();
|
|
(void) ZCancelSubscriptions(0);
|
|
#ifdef DEBUG
|
|
}
|
|
#endif /* DEBUG */
|
|
ZClosePort();
|
|
}
|
|
return;
|
|
}
|