[0001] [0002] [0003] [0004] [0005] [0006] [0007] [0008] [0009] [0010] [0011] [0012] [0013] [0014] [0015] [0016] [0017] [0018] [0019] [0020] [0021] [0022] [0023] [0024] [0025] [0026] [0027] [0028] [0029] [0030] [0031] [0032] [0033] [0034] [0035] [0036] [0037] [0038] [0039] [0040] [0041] [0042] [0043] [0044] [0045] [0046] [0047] [0048] [0049] [0050] [0051] [0052] [0053] [0054] [0055] [0056] [0057] [0058] [0059] [0060] [0061] [0062] [0063] [0064] [0065] [0066] [0067] [0068] [0069] [0070] [0071] [0072] [0073] [0074] [0075] [0076] [0077] [0078] [0079] [0080] [0081] [0082] [0083] [0084] [0085] [0086] [0087] [0088] [0089] [0090] [0091] [0092] [0093] [0094] [0095] [0096] [0097] [0098] [0099] [0100] [0101] [0102] [0103] [0104] [0105] [0106] [0107] [0108] [0109] [0110] [0111] [0112] [0113] [0114] [0115] [0116] [0117] [0118] [0119] [0120] [0121] [0122] [0123] [0124] [0125] [0126] [0127] [0128] [0129] [0130] [0131] [0132] [0133] [0134] [0135] [0136] [0137] [0138] [0139] [0140] [0141] [0142] [0143] [0144] [0145] [0146] [0147] [0148] [0149] [0150] [0151] [0152] [0153] [0154] [0155] [0156] [0157] [0158] [0159] [0160] [0161] [0162] [0163] [0164] [0165] [0166] [0167] [0168] [0169] [0170] [0171] [0172] [0173] [0174] [0175] [0176] [0177] [0178] [0179] [0180] [0181] [0182] [0183] [0184] [0185] [0186] [0187] [0188] [0189] [0190] [0191] [0192] [0193] [0194] [0195] [0196] [0197] [0198] [0199] [0200] [0201] [0202] [0203] [0204] [0205] [0206] [0207] [0208] [0209] [0210] [0211] [0212] [0213] [0214] [0215] [0216] [0217] [0218] [0219] [0220] [0221] [0222] [0223] [0224] [0225] [0226] [0227] [0228] [0229] [0230] [0231] [0232] [0233] [0234] [0235] [0236] [0237] [0238] [0239] [0240] [0241] [0242] [0243] [0244] [0245] [0246] [0247] [0248] [0249] [0250] [0251] [0252] [0253] [0254] [0255] [0256] [0257] [0258] [0259] [0260] [0261] [0262] [0263] [0264] [0265] [0266] [0267] [0268] [0269] [0270] [0271] [0272] [0273] [0274] [0275] [0276] [0277] [0278] [0279] [0280] [0281] [0282] [0283] [0284] [0285] [0286] [0287]
/* * Provide support routines for managing index file used for counting * accesses: * * void *ifopen ( char *fname, char *mode ); * int ifclose ( void *ifile ); * int iferror ( void *ifile ); * int ifread_rec ( char *buf, size_t bufsize, size_t *len, void *ifile, * char *key, int key_size ); * int ifwrite_rec ( char *buf, size_t bufsize, void *ifile ); * int ifupdate_rec ( char *buf, size_t bufsize, void *ifile ); * int ifdlcreate ( char *fld_str, char *fname, char *defname ); * int icheck_access ( char *fname, int uic ); * * Base on indexio.c originally supplied by Ken Covert. * * Revised: 23-APR-1995 Added ifdlcreate and icheck_access. (DLJ) * Revised: 24-APR-1995 Add defaul ACL if empty. * Revised: 25-APR-1995 Cleanup up RMS calls, delete unused routine. * Revised: 18-JAN-1996 Added ifdelete_rec function. */ #include <stdio.h> #include <rms.h> #include <string.h> #include <starlet.h> #include <lib$routines.h> #include <ssdef.h> #include <acldef.h> #include <armdef.h> #include <chpdef.h> #include <fdldef.h> #include <jpidef.h> #include <descrip.h> #include <stdlib.h> #include <stsdef.h> typedef struct { struct FAB fab; struct RAB rab; union { int rc; struct { unsigned severity:1; unsigned return_condition:31; } status_parts; } status; } IFILE; #define IFILE_T 1 #include "access_db.h" /*****************************************************************************/ /* Since ifdlcreate and icheck_access are called extremely rarely, dynamically * load FDLSHR and SECURESHR so we don't burden every run of the program * with 3 additional shareable image activations. */ static int (*fdl_create)(), (*sys_change_acl)(); static int images_loaded = 0; static int load_images() { $DESCRIPTOR(fdlshr,"FDLSHR"); $DESCRIPTOR(securshr,"SECURESHR"); $DESCRIPTOR(fdlcreate,"FDL$CREATE"); $DESCRIPTOR(change_acl,"SYS$CHANGE_ACL"); int status; status = lib$find_image_symbol ( &fdlshr, &fdlcreate, &fdl_create ); if ( (status&1) == 0 ) return status; status = lib$find_image_symbol ( &securshr, &change_acl, &sys_change_acl ); if ( (status&1) == 0 ) return status; images_loaded = 1; return status; } /****************************************************************************/ int ifdlcreate ( char *fdl, char *fname, char *defname ) { int status, flags; $DESCRIPTOR(fdl_desc,""); $DESCRIPTOR(filename,""); $DESCRIPTOR(default_name,""); /* * Set descriptors to point to passed strings and call FDL$CREATE. */ if ( !images_loaded ) load_images(); fdl_desc.dsc$w_length = strlen ( fdl ); fdl_desc.dsc$a_pointer = fdl; filename.dsc$w_length = strlen ( fname ); filename.dsc$a_pointer = fname; default_name.dsc$w_length = strlen(defname); default_name.dsc$a_pointer = defname; flags = FDL$M_FDL_STRING | FDL$M_SIGNAL; status = fdl_create (&fdl_desc, &filename, &default_name, 0, 0, &flags); return status; } /****************************************************************************/ /* Retrieve ACL on indicated filename and see if it permits read access * to specified UIC. */ int icheck_access ( char *fname, int uic ) { struct { short len, code; char *buffer; int *retlen; } itemlist[10]; #define SETITEM(itm,a,b,c,d) \ itm.len=a; itm.code=b; itm.buffer=(char *) c; itm.retlen=(int *) d; int status, acllen, context, objtype; long access, rights[2], privs[2]; char *acl; $DESCRIPTOR(filename,""); /* * Get length of ACL. */ if ( !images_loaded ) load_images(); SETITEM(itemlist[0],4,ACL$C_ACLLENGTH, &acllen, 0) itemlist[1].len = itemlist[1].code = 0; objtype = ACL$C_FILE; context = 0; filename.dsc$w_length = strlen ( fname ); filename.dsc$a_pointer = fname; status = (*sys_change_acl)( 0, &objtype, &filename, &itemlist, 0, 0, 0, 0, 0 ); if ( (status&1) == 0 ) return status; /* * Allocate buffer and read entire ACL. */ acl = malloc ( acllen ); if ( !acl ) return 12; SETITEM(itemlist[0],acllen,ACL$C_READACL,acl,&acllen) status = (*sys_change_acl)( 0, &objtype, &filename, &itemlist, 0, 0, 0, 0, 0 ); if ( status == SS$_ACLEMPTY ) { /* * Make phony ACL granting access to current process UIC. */ int iosb[2], pid, proc_uic; SETITEM(itemlist[0],4,JPI$_UIC,&proc_uic, 0); itemlist[1].len = itemlist[1].code = 0; /* null list */ pid = 0; status = sys$getjpiw ( 0, &pid, 0, &itemlist, iosb, 0, 0 ); if ( (status&1) == 1 ) status = iosb[0]; if ( (status&1) == 1 ) { if ( proc_uic != uic ) status = SS$_NOPRIV; } } else if ( (status&1) == 1 ) { /* * We have ACL, construct item list for $CHKPRO. */ access = ARM$M_READ; rights[0] = uic; rights[1] = 0; privs[0] = privs[1] = 0; SETITEM(itemlist[0],4,CHP$_ACCESS,&access,0) SETITEM(itemlist[1],8,CHP$_RIGHTS,rights,0) SETITEM(itemlist[2],8,CHP$_PRIV,privs,0) SETITEM(itemlist[3],acllen,CHP$_ACL,acl,0) itemlist[4].len = itemlist[4].code = 0; status = sys$chkpro ( &itemlist ); } free ( acl ); return status; } /****************************************************************************/ IFILE* ifopen(char *fn, char *mode) { IFILE *ifile; ifile = malloc(sizeof(IFILE)); ifile->fab = cc$rms_fab; if (strcmp(mode,"r") == 0) ifile->fab.fab$b_fac = FAB$M_GET; else if (strcmp(mode,"r+") == 0) ifile->fab.fab$b_fac = FAB$M_DEL | FAB$M_GET | FAB$M_PUT | FAB$M_UPD; else { fprintf(stderr,"I-file open error - invalid mode \"%s\"",mode); exit(0); } ifile->fab.fab$l_fna = fn; ifile->fab.fab$b_fns = strlen(fn); ifile->fab.fab$b_shr = FAB$V_MSE | FAB$V_SHRPUT | FAB$V_SHRGET | FAB$V_SHRDEL | FAB$V_SHRUPD; ifile->status.rc = sys$open(&(ifile->fab)); if (ifile->status.status_parts.severity != STS$K_SUCCESS) { free(ifile); return(NULL); } ifile->rab = cc$rms_rab; ifile->rab.rab$l_fab = &(ifile->fab); ifile->status.rc = sys$connect(&(ifile->rab)); if (ifile->status.status_parts.severity != STS$K_SUCCESS) { free(ifile); return(NULL); } return ifile; } /****************************************************************************/ int ifclose(IFILE *ifile) { ifile->status.rc = sys$close(&(ifile->fab)); if (ifile->status.status_parts.severity != STS$K_SUCCESS) return EOF; free(ifile); return 0; } /****************************************************************************/ /* Check status of last ifile call and signal error. */ int iferror(IFILE *ifile) { if (ifile->status.status_parts.severity == STS$K_SUCCESS) return 0; else { lib$signal(ifile->status.rc); return ifile->status.rc; } } /****************************************************************************/ /* Read keyed record from input file, locking for update. */ int ifread_rec(void *ptr, size_t size_of_item, size_t *size_read, IFILE *ifile, int index, void *key, int key_size) { int i; if (key_size==0) { ifile->rab.rab$b_rac = RAB$C_SEQ; ifile->rab.rab$l_rop = RAB$M_WAT; } else { ifile->rab.rab$b_rac = RAB$C_KEY; ifile->rab.rab$l_rop = RAB$M_WAT | RAB$M_RLK; ifile->rab.rab$l_kbf = key; ifile->rab.rab$b_ksz = key_size; } ifile->rab.rab$l_ubf = ptr; ifile->rab.rab$w_usz = size_of_item; ifile->rab.rab$b_krf = index; ifile->status.rc = sys$get(&(ifile->rab)); if (ifile->status.status_parts.severity == STS$K_SUCCESS) { *size_read = ifile->rab.rab$w_rsz; } else { *size_read = 0; } return ifile->status.rc; } /****************************************************************************/ /* write indexed file record. Return value is VMS status of $UPDATE call. */ int ifwrite_rec(void *ptr, size_t size_of_item, IFILE *ifile) { ifile->rab.rab$b_rac = RAB$C_KEY; ifile->rab.rab$l_rbf = ptr; ifile->rab.rab$w_rsz = size_of_item; ifile->rab.rab$l_rop = RAB$M_UIF | RAB$M_WAT; ifile->status.rc = sys$put(&(ifile->rab)); return ifile->status.rc; } /****************************************************************************/ /* Update indexed file record. Return value is VMS status of $UPDATE call. */ int ifupdate_rec(void *ptr, size_t size_of_item, IFILE *ifile) { ifile->rab.rab$b_rac = RAB$C_KEY; ifile->rab.rab$l_rbf = ptr; ifile->rab.rab$w_rsz = size_of_item; ifile->rab.rab$l_rop = RAB$M_UIF | RAB$M_WAT; ifile->status.rc = sys$update(&(ifile->rab)); return ifile->status.rc; } /****************************************************************************/ /* Delete current file record. Return value is VMS status of $DELETE call. */ int ifdelete_rec( IFILE *ifile ) { int sys$delete(); ifile->rab.rab$l_rop = 0; ifile->status.rc = sys$delete(&(ifile->rab)); return ifile->status.rc; }