/* ssaprelease1.c - SPM: initiate release */
static char *rcsid
= "$Header: /f/osi/ssap/RCS/ssaprelease1.c,v 7.1 91/02/22 09:45:56 mrose Interim $";
* $Header: /f/osi/ssap/RCS/ssaprelease1.c,v 7.1 91/02/22 09:45:56 mrose Interim $
* $Log: ssaprelease1.c,v $
* Revision 7.1 91/02/22 09:45:56 mrose
* Revision 7.0 89/11/23 22:25:34 mrose
* Acquisition, use, and distribution of this module and related
* materials are subject to the restrictions of a license agreement.
* Consult the Preface in the User's Manual for the full terms of
/* \f S-RELEASE.REQUEST */
int SRelRequest (sd
, data
, cc
, secs
, sr
, si
)
struct SSAPindication
*si
;
register struct ssapblk
*sb
;
toomuchP (sb
, data
, cc
, SF_SIZE
, "release");
result
= SRelRequestAux (sb
, data
, cc
, secs
, sr
, si
);
(void) sigiomask (smask
);
#define dotoken(requires,shift,bit,type) \
if ((sb -> sb_requirements & requires) && !(sb -> sb_owned & bit)) \
return ssaplose (si, SC_OPERATION, NULLCP, \
"%s token not owned by you", type); \
static int SRelRequestAux (sb
, data
, cc
, secs
, sr
, si
)
register struct ssapblk
*sb
;
struct SSAPindication
*si
;
register struct ssapkt
*s
;
if (sb
-> sb_flags
& SB_CD
)
return ssaplose (si
, SC_OPERATION
, NULLCP
,
"capability data request in progress");
if (sb
-> sb_flags
& SB_CDA
)
return ssaplose (si
, SC_OPERATION
, NULLCP
,
"awaiting your capability data response");
if (sb
-> sb_flags
& SB_GTC
)
return ssaplose (si
, SC_OPERATION
, NULLCP
,
"give control request in progress");
if (sb
-> sb_flags
& SB_MAA
)
return ssaplose (si
, SC_OPERATION
, "awaiting your majorsync response");
if (sb
-> sb_flags
& SB_RELEASE
)
return ssaplose (si
, SC_OPERATION
, "release already in progress");
if (sb
-> sb_xspdu
|| sb
-> sb_spdu
)
return ssaplose (si
, SC_WAITING
, NULLCP
, NULLCP
);
if ((s
= newspkt (SPDU_FN
)) == NULL
)
return ssaplose (si
, SC_CONGEST
, NULLCP
, "out of memory");
s
-> s_mask
|= SMASK_UDATA_PGI
;
s
-> s_udata
= data
, s
-> s_ulen
= cc
;
s
-> s_udata
= NULL
, s
-> s_ulen
= 0;
return SRelRetryRequestAux (sb
, secs
, sr
, si
);
/* \f S-RELEASE-RETRY.REQUEST (pseudo) */
int SRelRetryRequest (sd
, secs
, sr
, si
)
struct SSAPindication
*si
;
register struct ssapblk
*sb
;
if ((sb
= findsblk (sd
)) == NULL
)
result
= ssaplose (si
, SC_PARAMETER
, NULLCP
,
"invalid session descriptor");
if (!(sb
-> sb_flags
& SB_RELEASE
))
result
= ssaplose (si
, SC_OPERATION
, "release not in progress");
result
= SRelRetryRequestAux (sb
, secs
, sr
, si
);
(void) sigiomask (smask
);
static int SRelRetryRequestAux (sb
, secs
, sr
, si
)
register struct ssapblk
*sb
;
struct SSAPindication
*si
;
register struct ssapkt
*s
;
if (sb
-> sb_flags
& SB_RELEASE
)
if (((s
= sb
-> sb_retry
) -> s_code
= code
) == SPDU_FN
) {
s
-> s_mask
|= SMASK_FN_DISC
;
s
-> s_fn_disconnect
= FN_DISC_RELEASE
;
result
= spkt2sd (s
, sb
-> sb_fd
, 0, si
);
if (s
-> s_code
== SPDU_FN
) {
s
-> s_mask
&= ~(SMASK_UDATA_PGI
| SMASK_FN_DISC
);
s
-> s_udata
= NULL
, s
-> s_ulen
= 0;
s
-> s_fn_disconnect
= 0;
if ((s
= sb2spkt (sb
, si
, secs
, NULLTX
)) == NULL
) {
register struct SSAPabort
*sa
= &si
-> si_abort
;
if (sa
-> sa_reason
== SC_TIMER
) {
sb
-> sb_flags
|= SB_RELEASE
;
bzero ((char *) sr
, sizeof *sr
);
sr
-> sr_affirmative
= 1;
if (!(sb
-> sb_requirements
& SR_RLS_EXISTS
)
|| !(sb
-> sb_owned
& ST_RLS_TOKEN
))
sr
-> sr_affirmative
= 0;
if (sb
-> sb_spdu
) /* XXX */
freespkt (sb
-> sb_spdu
);
return ssaplose (si
, SC_WAITING
, NULLCP
, NULLCP
);
si
-> si_type
= SI_ABORT
;
register struct SSAPabort
*sa
= &si
-> si_abort
;
if (!(sa
-> sa_peer
= (s
-> s_ab_disconnect
& AB_DISC_USER
)
sa
-> sa_reason
= SC_ABORT
;
sa
-> sa_info
= s
-> s_udata
, sa
-> sa_cc
= s
-> s_ulen
;
sa
-> sa_realinfo
= s
-> s_udata
, s
-> s_udata
= NULL
;
(void) spktlose (sb
-> sb_fd
, si
, SC_PROTOCOL
, NULLCP
,
"session protocol mangled: not expecting 0x%x",