DataMuseum.dk

Presents historical artifacts from the history of:

DKUUG/EUUG Conference tapes

This is an automatic "excavation" of a thematic subset of
artifacts from Datamuseum.dk's BitArchive.

See our Wiki for more about DKUUG/EUUG Conference tapes

Excavated with: AutoArchaeologist - Free & Open Source Software.


top - metrics - download
Index: T c

⟦b0acb4b94⟧ TextFile

    Length: 10031 (0x272f)
    Types: TextFile
    Names: »chan_control.c«

Derivation

└─⟦2d1937cfd⟧ Bits:30007241 EUUGD22: P.P 5.0
    └─⟦dc59850a2⟧ »EurOpenD22/pp5.0/pp-5.tar.Z« 
        └─⟦e5a54fb17⟧ 
            └─⟦this⟧ »pp-5.0/Lib/qmgr/chan_control.c« 

TextFile

/* channel_control.c: control the channel process */

# ifndef lint
static char Rcsid[] = "@(#)$Header: /cs/research/pp/hubris/pp-beta/Lib/qmgr/RCS/chan_control.c,v 5.0 90/09/20 16:22:05 pp Exp Locker: pp $";
# endif

/*
 * $Header: /cs/research/pp/hubris/pp-beta/Lib/qmgr/RCS/chan_control.c,v 5.0 90/09/20 16:22:05 pp Exp Locker: pp $
 *
 * $Log:	chan_control.c,v $
 * Revision 5.0  90/09/20  16:22:05  pp
 * rcsforce : 5.0 public release
 * 
 */



#include <stdio.h>
#include <syslog.h>
#include <setjmp.h>
#include <varargs.h>
#include <isode/rosy.h>
#include <isode/tsap.h>         /* for listening */
#include "qmgr.h"
#include "retcode.h"
#include "ll_log.h"

static jmp_buf toplevel;


static IFP      startfnx;
static IFP      stopfnx;
static IFP      initchfnx;
static struct type_Qmgr_DeliveryStatus *(*workfnx)();

static int     ros_init (), ros_work (), ros_lose ();
static int     ros_worker (), error (), ureject ();
static void    adios ();
static void    acs_advise ();
static void    ros_adios (), ros_advise ();
static void    ros_indication ();

#ifdef notdef
static  char    *myservice = "pp channel";
#endif
extern struct RyOperation table_QmgrChannel_Operations[];

int channel_control (argc, argv, init, work, finish)
int     argc;
char    **argv;
IFP     init;
struct type_Qmgr_DeliveryStatus *(*work)();
IFP     finish;
{
	AEI         aei = NULLAEI;
	struct TSAPdisconnect   tds;
	struct TSAPdisconnect  *td = &tds;
	struct RoSAPindication  rois;
	register struct RoSAPindication *roi = &rois;
	register struct RoSAPpreject   *rop = &roi -> roi_preject;

	PP_DBG (("starting"));

	workfnx = work;
	initchfnx = init;
	stopfnx = finish;

	if (RyDispatch (NOTOK, table_QmgrChannel_Operations,
			operation_Qmgr_processmessage,
			ros_worker, roi) == NOTOK)
		ros_adios (rop, "processmessage");
	if (RyDispatch (NOTOK, table_QmgrChannel_Operations,
			operation_Qmgr_channelInitialise,
			ros_worker, roi) == NOTOK)
		ros_adios (rop, "channelInitialise");

	if (isodeserver (argc, argv, aei, ros_init, ros_work, ros_lose, td)
	    == NOTOK) {
		if (td -> td_cc > 0)
			adios (NULLCP, "isodeserver: [%s] %*.*s",
			       TErrString (td -> td_reason),
			       td -> td_cc, td -> td_cc,
			       td -> td_data);
		else
			adios (NULLCP, "isodeserver: [%s]",
			       TErrString (td -> td_reason));
	}

	return 0;
}

static int ros_result (sd, val, rox, roi)
int     sd;
struct type_Qmgr_DeliveryStatus *val;
struct RoSAPinvoke *rox;
struct RoSAPindication *roi;
{
	if (val == NULL)
		return error (sd, error_Qmgr_congested, (caddr_t) NULL,
			      rox, roi);

	if (RyDsResult (sd, rox -> rox_id, (caddr_t) val, ROS_NOPRIO, roi)
			== NOTOK)
		ros_adios (&roi -> roi_preject, "RESULT");
	return OK;
}

static int ros_worker (sd, ryo, rox, in, roi)
int     sd;
struct RyOperation *ryo;
struct RoSAPinvoke *rox;
caddr_t in;
struct RoSAPindication *roi;
{
	struct type_Qmgr_DeliveryStatus *status;

	if (rox -> rox_nolinked == 0) {
		PP_LOG (LLOG_EXCEPTIONS,
			("RO-INVOKE.INDICATION/%d: %s, unknown linkage %d",
			sd, ryo -> ryo_name, rox -> rox_linkid));
		return ureject (sd, ROS_IP_LINKED, rox, roi);
	}
	PP_DBG (("RO-INVOKE.INDICATION/%d: %s", sd, ryo -> ryo_name));

	switch (ryo -> ryo_op) {
	    case operation_Qmgr_channelInitialise:
		switch ((*initchfnx) (in)) {
		    case OK:
			if (RyDsResult (sd, rox -> rox_id, (caddr_t) NULL,
					ROS_NOPRIO, roi) == NOTOK)
				ros_adios (&roi -> roi_preject, "RESULT");
			return OK;

		    case NOTOK:
		    default:
			return error (sd, error_Qmgr_protocol, (caddr_t) NULL,
				      rox, roi);
		}

	    case operation_Qmgr_processmessage:
		status = (*workfnx) (in);
		return ros_result (sd, status, rox, roi);

	    default:
		PP_LOG (LLOG_EXCEPTIONS,
			("RO-INVOKE.INDICATION/%d: operation %s not expected",
			sd, ryo -> ryo_name));
		return error (sd, error_Qmgr_protocol, (caddr_t) NULL,
			      rox, roi);
	}
}

/* \f

 */

static int  ros_init (vecp, vec)
int     vecp;
char  **vec;
{
    int     reply,
	    result,
	    sd;
    struct AcSAPstart   acss;
    register struct AcSAPstart *acs = &acss;
    struct AcSAPindication  acis;
    register struct AcSAPindication *aci = &acis;
    register struct AcSAPabort   *aca = &aci -> aci_abort;
    register struct PSAPstart *ps = &acs -> acs_start;
    struct RoSAPindication  rois;
    register struct RoSAPindication *roi = &rois;
    register struct RoSAPpreject   *rop = &roi -> roi_preject;

    if (AcInit (vecp, vec, acs, aci) == NOTOK) {
	acs_advise (aca, "initialization fails");
	return NOTOK;
    }
    PP_DBG (("A-ASSOCIATE.INDICATION: <%d, %s, %s, %s, %d>",
		acs -> acs_sd, sprintoid (acs -> acs_context),
		sprintaei (&acs -> acs_callingtitle),
		sprintaei (&acs -> acs_calledtitle), acs -> acs_ninfo));

    sd = acs -> acs_sd;

    for (vec++; *vec; vec++)
	PP_LOG (LLOG_EXCEPTIONS, ("unknown argument \"%s\"", *vec));

    reply = startfnx ? (*startfnx) (sd, acs) : ACS_ACCEPT;

    result = AcAssocResponse (sd, reply, 
		reply != ACS_ACCEPT ? ACS_USER_NOREASON : ACS_USER_NULL,
		NULLOID, NULLAEI, NULLPA, NULLPC, ps -> ps_defctxresult,
		ps -> ps_prequirements, ps -> ps_srequirements, SERIAL_NONE,
		ps -> ps_settings, &ps -> ps_connect, NULLPEP, 0, aci);

    ACSFREE (acs);

    if (result == NOTOK) {
	acs_advise (aca, "A-ASSOCIATE.RESPONSE");
	return NOTOK;
}
    if (reply != ACS_ACCEPT)
	return NOTOK;

    if (RoSetService (sd, RoPService, roi) == NOTOK)
	ros_adios (rop, "set RO/PS fails");

    return sd;
}

/* \f

 */

static int  ros_work (fd)
int     fd;
{
    int     result;
    caddr_t out;
    struct AcSAPindication  acis;
    struct RoSAPindication  rois;
    register struct RoSAPindication *roi = &rois;
    register struct RoSAPpreject   *rop = &roi -> roi_preject;

    switch (setjmp (toplevel)) {
	case OK: 
	    break;

	default: 
	    if (stopfnx)
		(*stopfnx) ();
	case DONE:
	    (void) AcUAbortRequest (fd, NULLPEP, 0, &acis);
	    (void) RyLose (fd, roi);
	    return NOTOK;
    }

    switch (result = RyWait (fd, NULLIP, &out, OK, roi)) {
	case NOTOK: 
	    if (rop -> rop_reason == ROS_TIMER)
		break;
	case OK: 
	case DONE: 
	    ros_indication (fd, roi);
	    break;

	default: 
	    adios (NULLCP, "unknown return from RoWaitRequest=%d", result);
    }

    return OK;
}

/* \f

 */

static void ros_indication (sd, roi)
int     sd;
register struct RoSAPindication *roi;
{
    int     reply,
	    result;

    switch (roi -> roi_type) {
	case ROI_INVOKE: 
	case ROI_RESULT: 
	case ROI_ERROR: 
	    adios (NULLCP, "unexpected indication type=%d", roi -> roi_type);
	    break;

	case ROI_UREJECT: 
	    {
		register struct RoSAPureject   *rou = &roi -> roi_ureject;

		if (rou -> rou_noid)
		    PP_LOG (LLOG_EXCEPTIONS, ("RO-REJECT-U.INDICATION/%d: %s",
			    sd, RoErrString (rou -> rou_reason)));
		else
		    PP_LOG (LLOG_EXCEPTIONS, 
			    ("RO-REJECT-U.INDICATION/%d: %s (id=%d)",
			    sd, RoErrString (rou -> rou_reason),
			    rou -> rou_id));
	    }
	    break;

	case ROI_PREJECT: 
	    {
		register struct RoSAPpreject   *rop = &roi -> roi_preject;

		if (ROS_FATAL (rop -> rop_reason))
		    ros_adios (rop, "RO-REJECT-P.INDICATION");
		ros_advise (rop, "RO-REJECT-P.INDICATION");
	    }
	    break;

	case ROI_FINISH: 
	    {
		register struct AcSAPfinish *acf = &roi -> roi_finish;
		struct AcSAPindication  acis;
		register struct AcSAPabort *aca = &acis.aci_abort;

		PP_TRACE (("A-RELEASE.INDICATION/%d: %d",
			sd, acf -> acf_reason));

		result = AcRelResponse (sd, reply = ACS_ACCEPT,
					ACR_NORMAL, NULLPEP, 0, &acis);

		ACFFREE (acf);

		if (result == NOTOK)
		    acs_advise (aca, "A-RELEASE.RESPONSE");
		else
		    if (reply != ACS_ACCEPT)
			break;

		if (stopfnx)
			(*stopfnx)();

		longjmp (toplevel, DONE);
	    }
	/* NOTREACHED */

	default: 
	    adios (NULLCP, "unknown indication type=%d", roi -> roi_type);
    }
}

/* \f

 */

static int  ros_lose (td)
struct TSAPdisconnect *td;
{
    if (td -> td_cc > 0)
	PP_LOG (LLOG_EXCEPTIONS, ("TNetAccept: [%s] %*.*s",
		TErrString (td -> td_reason), td -> td_cc, td -> td_cc,
		td -> td_data));
    else
	PP_LOG (LLOG_EXCEPTIONS, ("TNetAccept: [%s]",
		TErrString (td -> td_reason)));
}

/* \f

   ERRORS */

static void    ros_adios (rop, event)
register struct RoSAPpreject *rop;
char   *event;
{
    ros_advise (rop, event);

    longjmp (toplevel, NOTOK);
}


static void    ros_advise (rop, event)
register struct RoSAPpreject *rop;
char   *event;
{
    char    buffer[BUFSIZ];

    if (rop -> rop_cc > 0)
	(void) sprintf (buffer, "[%s] %*.*s", RoErrString (rop -> rop_reason),
		rop -> rop_cc, rop -> rop_cc, rop -> rop_data);
    else
	(void) sprintf (buffer, "[%s]", RoErrString (rop -> rop_reason));

    PP_LOG (LLOG_EXCEPTIONS, ("%s: %s", event, buffer));
}

/* \f

 */

static void    acs_advise (aca, event)
register struct AcSAPabort *aca;
char   *event;
{
    char    buffer[BUFSIZ];

    if (aca -> aca_cc > 0)
	(void) sprintf (buffer, "[%s] %*.*s",
		AcErrString (aca -> aca_reason),
		aca -> aca_cc, aca -> aca_cc, aca -> aca_data);
    else
	(void) sprintf (buffer, "[%s]", AcErrString (aca -> aca_reason));

    PP_LOG (LLOG_EXCEPTIONS, ("%s: %s (source %d)", event, buffer,
		aca -> aca_source));
}

/* \f

 */

#ifndef lint

static void    adios (va_alist)
va_dcl
{
    va_list ap;

    va_start (ap);

    _ll_log (pp_log_oper, LLOG_FATAL, ap);

    va_end (ap);

    _exit (1);
}
#else
/* VARARGS */

static void    adios (what, fmt)
char   *what,
       *fmt;
{
    adios (what, fmt);
}
#endif

/* \f

   ERROR */

static int  error (sd, err, param, rox, roi)
int     sd,
	err;
caddr_t param;
struct RoSAPinvoke *rox;
struct RoSAPindication *roi;
{
    if (RyDsError (sd, rox -> rox_id, err, param, ROS_NOPRIO, roi) == NOTOK)
	ros_adios (&roi -> roi_preject, "ERROR");

    return OK;
}

static int  ureject (sd, reason, rox, roi)
int     sd,
	reason;
struct RoSAPinvoke *rox;
struct RoSAPindication *roi;
{
    if (RyDsUReject (sd, rox -> rox_id, reason, ROS_NOPRIO, roi) == NOTOK)
	ros_adios (&roi -> roi_preject, "U-REJECT");

    return OK;
}