Google
 

Trailing-Edge - PDP-10 Archives - tops20-v7-ft-dist1-clock - 7-sources/diudir.b36
There are 4 other files named diudir.b36 in the archive. Click here to see a list.
%TITLE 'Data Interchange Utility DIRECTORY module'

MODULE DIUDIR (IDENT = '272',
               LANGUAGE(BLISS36),
               ENTRY(DIU$DIRECTORY)) =
BEGIN

!	COPYRIGHT (C) DIGITAL EQUIPMENT CORPORATION 1986, 1987.
!	ALL RIGHTS RESERVED.
!
!	THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY  BE  USED  AND
!	COPIED ONLY IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE AND WITH
!	THE INCLUSION OF THE ABOVE COPYRIGHT NOTICE.   THIS  SOFTWARE  OR
!	ANY  OTHER  COPIES  THEREOF MAY NOT BE PROVIDED OR OTHERWISE MADE
!	AVAILABLE TO ANY OTHER PERSON.  NO TITLE TO AND OWNERSHIP OF  THE
!	SOFTWARE IS HEREBY TRANSFERRED.
!
!	THE INFORMATION IN THIS SOFTWARE IS  SUBJECT  TO  CHANGE  WITHOUT
!	NOTICE  AND  SHOULD  NOT  BE CONSTRUED AS A COMMITMENT BY DIGITAL
!	EQUIPMENT CORPORATION.
!
!	DIGITAL ASSUMES NO RESPONSIBILITY FOR THE USE OR  RELIABILITY  OF
!	ITS SOFTWARE ON EQUIPMENT THAT IS NOT SUPPLIED BY DIGITAL.
!
! FACILITY:	DIU    Data Interchange Utility
!
! ENVIRONMENT:  TOPS-20 V6.1    XPORT      RMS V3
!		BLISS-36 V4     FAO-36
!
! ABSTRACT:     This module contains routines to perform file directory
!               displays. They will look at a set of RMS data structures 
!               for a file, and dump a directory listing to the passed rab.
!
! AUTHOR:  Rick Fricchione			CREATED: 7-Dec-1984
%SBTTL 'Revision History'
! HISTORY:
!
!  272	In DIUDIR.B36: Add new file system types.
!	Gregory A. Scott 6-Mar-87
!
!  271  In DIUDIR.B36: Add protection XAB and output of protections on /FULL.
!       Gregory A. Scott  16-Oct-86
!
!  252	Remove library of CONDIT.
!	Sandy Clemens  1-Jul-86
!
!  234  Change library of RMSUSR to RMSINT.
!       Gregory A. Scott 17-Jul-86
!
!  207  DIR$INITIALIZE did a crude job of ASCIZing the input spec.
!       Gregory A. Scott 27-May-86
!
!  206  Remove code from DIUDO that opened the  files and so forth and move  it
!       here for simplicity (module is now entered from DIUDO at  DIU$DIRECTORY
!       rather than  D$DIR).  Be  smarter about  the grand  total message  (say
!       pages and blocks if that is the case [yuk]).  Remove DIR$OPEN_OUTPUT.
!       Gregory A. Scott 27-May-86
!
!  177  Add call to S$BREATHE  in  inner  file  processing  loop.   Compilation
!       /DEBUG didn't work  because the  BIND to  get around  compiler bug  was
!       after the $TRACE in DIR$NORMAL.
!       Gregory A. Scott 22-May-86
!
!  176  Clean up DIR$_OPEN_OUTPUT, much of it is never used.   Implement  grand
!       total file size count.  Output long filenames properly.  Clean  up  the
!       rather gross implementation of $PUT_FAO.
!       Gregory A. Scott 21-May-86
!
!  164  Remove UNDECLARE of $PUT_FAO since it is no longer  in DIU.R36,  remove
!       cell fal_len since it is never used anymore (pass a 0 to $FAO instead).
!       Gregory A. Scott 16-May-86
!
!  131  Remove usage  fo diudbg  cell, replace  it with  conditional  assembly.
!       Replace macro $DEBUG_FAO with $TRACE_FAO and $DEBUG_TRACE with  $TRACE,
!       remove macro $CRLF since it was unreferenced.  Not entirely  clear  why
!       this module has a private $PUT_FAO macro; maybe someday I'll fix it.
!       Gregory A. Scott 28-Mar-86
!
!      55 Add type block of typ=image and change DIR$FULL to get
!         display of byte size, format, and record size
!         Andy Puchrik  2-Dec-85
!
!      40 Put the REQUIRE/LIBRARY of 'TOPS20' into a TOPS-20 only
!         conditional.
!         Sandy Clemens  7-Oct-85
!
!	RDF0001	V01-000		Rick Fricchione		7-Dec-1984
!		Original version of D$DIR.  Interface with $FAO
!               for pretty output.  Handle different levels of 
!               verbosity.
!
!
!--
%SBTTL 'Libraries and Externals'

! Libraries

LIBRARY 'BLI:XPORT';                    ! XPORT of course
LIBRARY 'FAO';                          ! FAO services
LIBRARY 'RMSINT';                       ! RMS services
LIBRARY 'DIU';                          ! DIU Data Structures

! Externals

EXTERNAL ROUTINE DO$BYPASS,             ! Bypass something do to with RMS
                 S$BREATHE : NOVALUE,   ! Let spooler take a breath
                 RMS$FAILURE,           ! RMS condition handler
                 RMS$SIGNAL;            ! RMS error handler
%SBTTL 'Forward Routine'

FORWARD ROUTINE
        DIU$DIRECTORY,                  ! Main entry point
        DIR$DO : NOVALUE,               ! Called for each source spec
        DIR$TOTAL : NOVALUE,            ! Print totals for this dir
        DIR$GRAND_TOTAL : NOVALUE,      ! Print total for this request
        DIR$BREAK_CHECK : NOVALUE,      ! Print header
        DIR$INITIALIZE : NOVALUE,       ! Initialize RMS data structures
        DIR$BRIEF : NOVALUE,            ! Show only filename
        DIR$NORMAL : NOVALUE,           ! Show a mediocre amount of data
        DIR$FULL : NOVALUE;             ! Show everything
%SBTTL 'Macros'

MACRO $PUT_FAO (control) = 
      BEGIN

      LOCAL fao_ctl : $STR_DESCRIPTOR(STRING=%STRING(control,%CHAR(13,10)));

      ! Init the FAO record descriptor

      $STR_DESC_INIT(DESC=fao_desc,
                     CLASS=BOUNDED,
                     STRING=(fao_buf_size, CH$PTR(fao_buf)));

      ! Call FAO to format the text 

      $FAO(fao_ctl,0,fao_desc,%REMAINING);

      ! Load the $RAB, write it to the file

      fao_rab[RAB$H_RSZ] = .fao_desc[STR$H_LENGTH];
      $PUT(RAB=.fao_rab,ERR=RMS$FAILURE);

      END%;

MACRO $APPEND_FAO (control) = 
      BEGIN

      LOCAL fao_ctl : $STR_DESCRIPTOR(STRING=control);

      ! Init the FAO record descriptor

      $STR_DESC_INIT(DESC=fao_desc,
                     CLASS=BOUNDED,
                     STRING=(fao_buf_size, CH$PTR(fao_buf)));

      ! Call FAO to format the text 

      $FAO(fao_ctl,0,fao_desc,%REMAINING);

      ! Load the $RAB, write it to the file

      fao_rab[RAB$H_RSZ] = .fao_desc[STR$H_LENGTH];
      $PUT(RAB=.fao_rab,ERR=RMS$FAILURE);

      END%;
%SBTTL 'Module Static Storage'

LITERAL fao_buf_size = 200;             ! FAO buffer size

OWN dir_files,                          ! Total files this dir
    dir_size,                           ! Total size this dir

    list_level,                         ! Directory listing level

    grand_blocks,                       ! Grand total blocks
    grand_pages,                        ! Grand total pages
    grand_files,                        ! Grand total files
    grand_dirs,                         ! Total directories

    dir_fab      : $FAB_DECL,                            ! Input file $FAB     
    dir_nam      : $NAM_DECL,                            ! Input file $NAM
    dir_dat      : $XABDAT_DECL,                         ! Input file $XABDAT 
    dir_cfg      : $XABCFG_DECL,                         ! Input file $XABCFG 
    dir_sum      : $XABSUM_DECL,                         ! Input file $XABSUM 
    dir_pro      : $XABPRO_DECL,                         ! Input file $XABPRO
    dir_typ      : $TYP_DECL,                            ! Input file $TYP

    filename_buf : VECTOR[CH$ALLOCATION(NAM$K_MAXRSS)],  ! $FAB ASCIZ buffer
    dir_esa      : VECTOR[CH$ALLOCATION(NAM$K_MAXRSS)],  ! $NAM expanded name
    dir_rsa      : VECTOR[CH$ALLOCATION(NAM$K_MAXRSS)],  ! $NAM resultant name

    sav_node     : VECTOR[CH$ALLOCATION(100)],           ! Saved node + access
    sav_dev      : VECTOR[CH$ALLOCATION(100)],           ! Saved device spec
    sav_dir      : VECTOR[CH$ALLOCATION(100)],           ! Saved directory
    nod_len      : INITIAL(0),                           ! Size of saved node
    dev_len      : INITIAL(0),                           ! Size of saved device
    dir_len      : INITIAL(0),                           ! Size of saved dir

    fao_rab      : REF $RAB_DECL,                        ! Output $RAB pointer
    fao_buf      : VECTOR[CH$ALLOCATION(fao_buf_size)],
    fao_desc     : $STR_DESCRIPTOR(CLASS=BOUNDED);       
%SBTTL 'Routine DIR$GRAND_TOTAL'

ROUTINE DIR$GRAND_TOTAL : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!       This routine gets called by us when the request has completed.
!
! IMPLICIT PARAMETERS
!
!	Module static storage used for directory file and size counters.
!
!--

$TRACE('DIR$GRAND_TOTAL');

IF .grand_dirs LEQ 1 AND .grand_files LEQ 1     ! Should we say anything?
THEN RETURN;                                    ! Nope

! If brief mode, then just list the number of files

IF .list_level EQL DIU$K_LIST_BRIEF
THEN BEGIN
     $PUT_FAO('!/  Grand total of !SL file!%S',
              .grand_files);
     RETURN;
     END;

! Output appropriate message based on if we have seen pages and/or blocks.

IF .grand_pages NEQ 0 AND .grand_blocks NEQ 0
THEN $PUT_FAO('!/  Grand total of !SL page!%S and !SL block!%S in !SL file!%S',
              .grand_pages,
              .grand_blocks,
              .grand_files)
ELSE IF .grand_blocks EQL 0
     THEN $PUT_FAO('!/  Grand total of !SL page!%S in !SL file!%S',
                   .grand_pages,
                   .grand_files)
     ELSE $PUT_FAO('!/  Grand total of !SL block!%S in !SL file!%S',
                   .grand_blocks,
                   .grand_files);
END;                                    ! DIR$GRAND_TOTAL
%SBTTL 'Routine DIR$TOTAL'

ROUTINE DIR$TOTAL : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!       This routine gets called by us when the directory has changed
!       (not the first time of course..) or we have finished our
!       work.  We then wish to display the total for the directory we
!       have just completed.
!
! IMPLICIT PARAMETERS
!
!	Module static storage used for directory file and size counters.
!
!--

! Output the directory total

IF .list_level NEQ DIU$K_LIST_BRIEF
THEN $PUT_FAO('!/  Total of !SL !AZ!%S in !SL file!%S',
              .dir_size,
              (IF ( (.dir_cfg[XAB$B_FILESYS] EQL XAB$K_FILESYS_TOPS20) OR
                  (.dir_cfg[XAB$B_FILESYS] EQL XAB$K_FILESYS_RMS20))
              THEN CH$PTR(UPLIT(%ASCIZ'page'))
              ELSE CH$PTR(UPLIT(%ASCIZ'block'))),
              .dir_files)
ELSE $PUT_FAO('!/  Total of !SL file!%S',.dir_files);

! Roll directory totals into grand total

grand_files = .grand_files + .dir_files;
IF .dir_cfg[XAB$B_FILESYS] EQL XAB$K_FILESYS_TOPS20     ! Is it some
    OR .dir_cfg[XAB$B_FILESYS] EQL XAB$K_FILESYS_RMS20  !  form of orange?
THEN grand_pages = .grand_pages + .dir_size     ! Yes, count orange pages (20)
ELSE grand_blocks = .grand_blocks + .dir_size;  ! No, count blocks (10, VMS)

! Zero out the directory totals

dir_files   = 0;
dir_size  = 0;

END;                                    ! DIR$TOTAL
%SBTTL 'Routine DIR$INITIALIZE'

ROUTINE DIR$INITIALIZE (p_file) : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!       Init the module static storage that we will use in this function.
!
! FORMAL PARAMETERS
!
!       p_file: Address of descriptor for filespec we are going to look up.
!
! IMPLICIT PARAMETERS
!
!	Module static storage defined above.
!
!--

BIND file = .p_file : $STR_DESCRIPTOR(CLASS=DYNAMIC);

! Initialize our counters for a repeat trip

dir_files = 0;                          ! Total files in this directory
dir_size = 0;                           ! Total file sizes in this directory

! Initialize the $FAB used for lookups

$FAB_INIT(FAB=dir_fab,     FOP=<NAM,DRJ>,     FAC=<GET>,    
          SHR=GET,         NAM=dir_nam,       XAB=dir_sum,
          TYP=dir_typ,     MRS=1,             FNA=filename_buf);

! Initialize the $NAM, $TYP, & XABS

$NAM_INIT(NAM=dir_nam,        RSA=CH$PTR(dir_rsa), RSS=NAM$k_MAXRSS,
                              ESA=CH$PTR(dir_esa), ESS=NAM$K_MAXRSS);

$TYP_INIT(TYP=dir_typ,        CLASS=Typ$k_Image);

! Initialize the directory $XABs and build chain

$XABSUM_INIT(XAB=dir_sum, NXT=dir_cfg);
$XABCFG_INIT(XAB=dir_cfg, NXT=dir_dat);
$XABDAT_INIT(XAB=dir_dat, NXT=dir_pro);
$XABPRO_INIT(XAB=dir_pro);

! Copy the given input spec over, make it ASCIZ so RMS likes it.

CH$COPY(.file[STR$H_LENGTH],.file[STR$A_POINTER],
        0,(.file[STR$H_LENGTH]+1),CH$PTR(filename_buf));

END;                                    ! DIR$INITIALIZE
%SBTTL 'Routine DIR$BREAK_CHECK'

ROUTINE DIR$BREAK_CHECK : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!       This routine will check the contents of the NAM block against
!       the last breakpoint displayed and print node::dev:[dir] information
!       on a seperate line.
!
!       The format of this line is 
!
!            LATOUR::DISK$DRACULA:[RDF]   (RMS-32)   or
!            KL2102::EXODUS:[FRICCHIONE]  (RMS-20)   or
!            EXODUS:[FRICCHIONE] (Local)
!
! FORMAL PARAMETERS
!
!       None
!
! IMPLICIT PARAMETERS
!
!       Module static storage
!
! ROUTINE VALUE
!
!       None.  It either prints something or doesn't.
!
!--

LOCAL rfs             : INITIAL(0),
      printing_header : INITIAL(FALSE);

$TRACE('DIR$BREAK_CHECK');

!$TRACE_FAO('Old Node: !AD!_ New Node: !AD',
!                      .nod_len,            CH$PTR(sav_node),
!                      .dir_nam[NAM$B_NODE], .dir_nam[NAM$A_NODE]);

!$TRACE_FAO('Old Dev:  !AD!_ New Dev:  !AD',
!                      .dev_len,             CH$PTR(sav_dev),
!                      .dir_nam[NAM$B_DEV],  .dir_nam[NAM$A_DEV]);

!$TRACE_FAO('Old Dir:  !AD!_ New Dir:  !AD',
!                      .dir_len,            CH$PTR(sav_dir),
!                      .dir_nam[NAM$B_DIR], .dir_nam[NAM$A_DIR]);

! Check the node name first to see if it has changed

IF CH$NEQ(.dir_nam[NAM$B_NODE],  .dir_nam[NAM$A_NODE],
          .nod_len,              CH$PTR(sav_node)) 
THEN printing_header = TRUE;

! Check the device

IF CH$NEQ(.dir_nam[NAM$B_DEV],   .dir_nam[NAM$A_DEV],
          .dev_len,              CH$PTR(sav_dev))
THEN printing_header = TRUE;

! Finally the directory

IF CH$NEQ(.dir_nam[NAM$B_DIR],   .dir_nam[NAM$A_DIR],
          .dir_len,              CH$PTR(sav_dir))
THEN printing_header = TRUE;

! If the file belongs to another directory, set up a break.

IF .printing_header
THEN BEGIN
     $TRACE('DIR$BREAK_CHECK Printing directory header');

     ! Increment count of how many we've run across

     grand_dirs = .grand_dirs + 1;

     ! Figure out who we are talking to

     SELECTONE .dir_cfg[XAB$B_FILESYS] OF
               SET
               [0]                      : rfs = CH$PTR(UPLIT(%ASCIZ'Local'));
               [XAB$K_FILESYS_RMS11]    : rfs = CH$PTR(UPLIT(%ASCIZ'RMS-11'));
               [XAB$K_FILESYS_RMS20]    : rfs = CH$PTR(UPLIT(%ASCIZ'RMS-20'));
               [XAB$K_FILESYS_RMS32]    : rfs = CH$PTR(UPLIT(%ASCIZ'RMS-32'));
               [XAB$K_FILESYS_FCS11]    : rfs = CH$PTR(UPLIT(%ASCIZ'FCS11'));
               [XAB$K_FILESYS_RT11]     : rfs = CH$PTR(UPLIT(%ASCIZ'RT-11'));
               [XAB$K_FILESYS_NONE]     : rfs = CH$PTR(UPLIT(%ASCIZ'None'));
               [XAB$K_FILESYS_TOPS20]   : rfs = CH$PTR(UPLIT(%ASCIZ'TOPS-20'));
               [XAB$K_FILESYS_TOPS10]   : rfs = CH$PTR(UPLIT(%ASCIZ'TOPS-10'));
               [XAB$K_FILESYS_OS8]      : rfs = CH$PTR(UPLIT(%ASCIZ'OS-8'));
               [XAB$K_FILESYS_RMS32S]   : rfs = CH$PTR(UPLIT(%ASCIZ'RMS-32S'));
               [XAB$K_FILESYS_CPM]      : rfs = CH$PTR(UPLIT(%ASCIZ'CP/M'));
               [XAB$K_FILESYS_MSDOS]    : rfs = CH$PTR(UPLIT(%ASCIZ'MSDOS'));
               [XAB$K_FILESYS_ULTRIX32] : rfs = CH$PTR(UPLIT(%ASCIZ'Ultrix-32'));
               [XAB$K_FILESYS_ULTRIX11] : rfs = CH$PTR(UPLIT(%ASCIZ'Ultrix-11'));
               [OTHERWISE]              : rfs = CH$PTR(UPLIT(%ASCIZ'Unknown'));
               TES;

         ! If not a remote filespec, make sure it reads local

         IF NOT .dir_fab[FAB$V_REMOTE]
         THEN rfs = CH$PTR(UPLIT(%ASCIZ'Local'));

         ! Print the directory totals if not first time through

         IF .dir_files NEQ 0
         THEN DIR$TOTAL();

         ! Display the line

         $PUT_FAO('!/!AD!AD!AD   (!AZ)!/',
                        .dir_nam[NAM$B_NODE],  .dir_nam[NAM$A_NODE],
                        .dir_nam[NAM$B_DEV],   .dir_nam[NAM$A_DEV],
                        .dir_nam[NAM$B_DIR],   .dir_nam[NAM$A_DIR],
                        .rfs);

         ! This is now our control break check

         nod_len = .dir_nam[NAM$B_NODE];
         dev_len  = .dir_nam[NAM$B_DEV];
         dir_len  = .dir_nam[NAM$B_DIR];
         CH$MOVE(.nod_len, .dir_nam[NAM$A_NODE],CH$PTR(sav_node));
         CH$MOVE(.dev_len, .dir_nam[NAM$A_DEV], CH$PTR(sav_dev));
         CH$MOVE(.dir_len, .dir_nam[NAM$A_DIR], CH$PTR(sav_dir));

         !$TRACE_FAO('Saved Node: !AD',.nod_len, CH$PTR(sav_node));
         !$TRACE_FAO('Saved Dev:  !AD',.dev_len, CH$PTR(sav_dev));
         !$TRACE_FAO('Saved Dir:  !AD',.dir_len, CH$PTR(sav_dir));

         END;

! Add these totals to running directory total

dir_files = .dir_files + 1;
dir_size  = .dir_size  + .dir_fab[FAB$G_ALQ];

END;                                    ! DIR$BREAK_CHECK
%SBTTL 'Routine DIR$BRIEF'

ROUTINE DIR$BRIEF : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!       Just display the filespec.  
!
! IMPLICIT PARAMETERS
!
!	$NAM block from module static storage
!
!--

$TRACE('DIR$BRIEF');

$PUT_FAO('  !AD!AD!AD',  .dir_nam[NAM$B_NAME],  .dir_nam[NAM$A_NAME],
                         .dir_nam[NAM$B_TYPE],  .dir_nam[NAM$A_TYPE],
                         .dir_nam[NAM$B_VER],   .dir_nam[NAM$A_VER]);

END;                                    ! DIR$BRIEF
%SBTTL 'Routine DIR$NORMAL'

ROUTINE DIR$NORMAL : NOVALUE  = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!         Display filespec info in the following format
!
!              GARK_FILE.GREG.33    21      12-JAN-1985 13:21:05
!
! IMPLICIT PARAMETERS
!
!	$NAM block from module static storage
!
!--

BIND compiler_bug_a = .dir_nam[NAM$A_NAME],     ! Get around BLISS v4 compiler
     compiler_bug_b = .dir_nam[NAM$B_NAME];

$TRACE('DIR$NORMAL');

! If a short file name (fits in 40 characters) then show the files normally

IF .dir_nam[NAM$B_NAME]+.dir_nam[NAM$B_TYPE]+.dir_nam[NAM$B_VER] LEQ 40
THEN $PUT_FAO('  !40<!AD!AD!AD!> !6SL   !%D',           ! Short file
!             .dir_nam[NAM$B_NAME],  .dir_nam[NAM$A_NAME],      ! File name
              compiler_bug_b,        compiler_bug_a,
              .dir_nam[NAM$B_TYPE],  .dir_nam[NAM$A_TYPE],      ! File type
              .dir_nam[NAM$B_VER],   .dir_nam[NAM$A_VER],       ! File version
              .dir_fab[FAB$G_ALQ],                              ! File size
              .dir_dat[XAB$G_RDT])                              ! Revision date
ELSE $PUT_FAO('  !AD!AD!AD!/!43* !6SL   !%D',        ! Long file
              .dir_nam[NAM$B_NAME],  .dir_nam[NAM$A_NAME],      ! File name
              .dir_nam[NAM$B_TYPE],  .dir_nam[NAM$A_TYPE],      ! File type
              .dir_nam[NAM$B_VER],   .dir_nam[NAM$A_VER],       ! File version
              .dir_fab[FAB$G_ALQ],                              ! File size
              .dir_dat[XAB$G_RDT]);                             ! Revision date

END;                                    ! DIR$NORMAL
%SBTTL 'Routine DIR$FULL'

ROUTINE DIR$FULL : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!         Display RMS filespec info in the following format
!
!         MARCH-PAYROLL.DAT.1         (Local)
!
!            Created:  1-Jan-1985 13:22:05     Size:   260 pages
!            Expires:  1-Sep-1985 13:22:05     
!            Revised:  1-Jan-1986 08:45:12   
!
!            Organization:      Indexed    Keys:        3   Areas:     1
!            Record Format:     Fixed      Bucket Size: 0   Byte Size: 7
!            Record Attributes: None       
!
! IMPLICIT PARAMETERS
!
!	$NAM block from module static storage
!
!--

LOCAL org : INITIAL(0),
      attributes : $STR_DESCRIPTOR(CLASS=DYNAMIC);

! Protection table for verbalizing the protection.  Note the DAP bit lit
! means that access is denied.

OWN pro_tab : VECTOR[16] INITIAL(%ASCII'RWED',   ! 00
                                 %ASCII'WED',    ! 01
                                 %ASCII'RED',    ! 02
                                 %ASCII'ED',     ! 03
                                 %ASCII'RWD',    ! 04
                                 %ASCII'WD',     ! 05
                                 %ASCII'RD',     ! 06
                                 %ASCII'D',      ! 07
                                 %ASCII'RWE',    ! 10
                                 %ASCII'WE',     ! 11
                                 %ASCII'RE',     ! 12
                                 %ASCII'E',      ! 13
                                 %ASCII'RW',     ! 14
                                 %ASCII'W',      ! 15
                                 %ASCII'R',      ! 16
                                 %ASCII'none');  ! 17

$TRACE('DIR$FULL');

$STR_DESC_INIT(DESC=attributes,CLASS=DYNAMIC);

! Print the filename, creation date/time, and the size of file

$PUT_FAO('  !AD!AD!AD!/  Created: !%D !_Size: !SL !AZ!%S',  
         .dir_nam[NAM$B_NAME], .dir_nam[NAM$A_NAME],
         .dir_nam[NAM$B_TYPE], .dir_nam[NAM$A_TYPE],
         .dir_nam[NAM$B_VER],  .dir_nam[NAM$A_VER],
         .dir_dat[XAB$G_CDT],
         .dir_fab[FAB$G_ALQ],
         (IF   (.dir_cfg[XAB$B_FILESYS] EQL XAB$K_FILESYS_TOPS20)
            OR (.dir_cfg[XAB$B_FILESYS] EQL XAB$K_FILESYS_RMS20)
          THEN CH$PTR(UPLIT(%ASCIZ'page'))
          ELSE CH$PTR(UPLIT(%ASCIZ'block'))));

! Output expiration and revision dates.

IF .dir_dat[XAB$G_EDT] NEQ 0
THEN $PUT_FAO('  Expires: !%D', .dir_dat[XAB$G_EDT]);

IF .dir_dat[XAB$G_RDT] NEQ 0
THEN $PUT_FAO('  Revised: !%D', .dir_dat[XAB$G_RDT]);

! Output protection codes like VMS does only if config says we got some there.

IF .dir_cfg[XAB$V_PROTECTION]
THEN $PUT_FAO('  File protection:   System: !AZ, Owner: !AZ, Group: !AZ, World: !AZ',
              CH$PTR(pro_tab[.dir_pro[XAB$V_PROTSYS] AND %O'17']),
              CH$PTR(pro_tab[.dir_pro[XAB$V_PROTOWN] AND %O'17']),
              CH$PTR(pro_tab[.dir_pro[XAB$V_PROTGRP] AND %O'17']),
              CH$PTR(pro_tab[.dir_pro[XAB$V_PROTWLD] AND %O'17']));

! Figure out what to print for the organization
! Print out the keys and areas only when dealing with indexed files

IF .dir_fab[FAB$V_ORG] EQL FAB$K_IDX
THEN $PUT_FAO('  Organization:      Indexed   !_Keys:        !4<!SB!> Areas:     !SB',
              .dir_sum[XAB$B_NOK],.dir_sum[XAB$B_NOA])
ELSE BEGIN
     BIND org=(SELECTONE .dir_fab[FAB$V_ORG] OF
                         SET
                         [0]         : CH$PTR(UPLIT(%ASCIZ'None'));
                         [FAB$K_SEQ] : CH$PTR(UPLIT(%ASCIZ'Sequential'));
                         [FAB$K_REL] : CH$PTR(UPLIT(%ASCIZ'Relative'));
                         [OTHERWISE] : CH$PTR(UPLIT(%ASCIZ'Unknown'));
                         TES);

      $PUT_FAO('  Organization:      !AZ',org);
      END;

! Print the record format, bucket size, byte size

IF .dir_fab[FAB$V_RFM] EQL FAB$K_UDF
THEN BEGIN
     SWITCHES LIST(EXPAND);
     $PUT_FAO('  Record Format:     Unknown            Byte Size: !SB',
              .dir_fab[FAB$V_BSZ]);
     END
ELSE BEGIN
     BIND recordformat=    ! Figure out what to print for the record format.
          (SELECTONE .dir_fab[FAB$V_RFM] OF
                     SET
                     [FAB$K_FIX] : CH$PTR(UPLIT(%ASCIZ'Fixed'));
                     [FAB$K_LSA] : CH$PTR(UPLIT(%ASCIZ'Line Seq'));
                     [FAB$K_VAR] : CH$PTR(UPLIT(%ASCIZ'Variable'));
                     [FAB$K_VFC] : CH$PTR(UPLIT(%ASCIZ'VFC'));
                     [FAB$K_STM] : CH$PTR(UPLIT(%ASCIZ'Stream'));
                     [OTHERWISE] : CH$PTR(UPLIT(%ASCIZ'Unknown'));
                     TES);

     $PUT_FAO('  Record Format:     !9<!AZ!>  Bucket Size: !4<!SW!> Byte Size: !SB',
              recordformat,
              .dir_fab[FAB$V_BKS],
              .dir_fab[FAB$V_BSZ]);

     ! Print the record size if any

     IF .dir_fab[FAB$H_MRS] NEQ 0
     THEN $PUT_FAO('  Maximum Record Size: !SW', .dir_fab[FAB$H_MRS] );

     ! Now print the record attributes

     IF .dir_fab[FAB$V_FTN]
     THEN $STR_APPEND(STRING=' Ftn',   TARGET=attributes);
     IF .dir_fab[FAB$V_BLK]
     THEN $STR_APPEND(STRING=' Blk',   TARGET=attributes);
     IF .dir_fab[FAB$V_CR]
     THEN $STR_APPEND(STRING=' Cr',    TARGET=attributes);
     IF .dir_fab[FAB$V_CBL]
     THEN $STR_APPEND(STRING=' Cbl',   TARGET=attributes);
     IF .dir_fab[FAB$V_MACY11]
     THEN $STR_APPEND(STRING=' MACY11',TARGET=attributes);
     IF .dir_fab[FAB$V_EMB]
     THEN $STR_APPEND(STRING=' Embedded', TARGET=attributes);
     IF .dir_fab[FAB$V_PRN]
     THEN $STR_APPEND(STRING=' Print', TARGET=attributes);

     IF .attributes[STR$H_LENGTH] EQL 0    ! Make sure it says something %$#&&%
     THEN $STR_COPY(STRING='None', TARGET=attributes);

     ! Print the record attributes

     $PUT_FAO('  Record Attributes: !AS',attributes);

     END;

! Output a blank line and then return

$PUT_FAO(' ');                          ! Be neat

END;                                    ! DIR$FULL
%SBTTL 'Routine DIR$DO'

GLOBAL ROUTINE DIR$DO (p_file) : NOVALUE = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION
!
!       Output the directory specified by p_file to the show_rab.
!
! FORMAL PARAMETERS
!
!       p_file: Address of a string descriptor indicating files to list.
!
! ROUTINE VALUE
!
!	DIU$_NORMAL		successful display of request block
!       DIU$_INVRMSBLK          invalid or incomplete RMS blocks passed
!
!--
BIND dir_file  = .p_file : $STR_DESCRIPTOR();

LABEL process;

LOCAL status;

DIR$INITIALIZE(dir_file);               ! Initialize module static storage

$PARSE(FAB=dir_fab,ERR=RMS$SIGNAL);     ! Set up wildcard context for the call

$TRACE('DIR$DO Parse of input file OK');

$SEARCH(FAB=dir_fab);                   ! Find the first file in the list 

UNTIL .dir_fab[FAB$H_STS] EQL RMS$_NMF  ! Loop for all files in the list
DO BEGIN
   process: BEGIN

            ! Check for RMS error (avoid NETSERVER.LOG conflicts)

            IF NOT $RMS_STATUS_OK(dir_fab)      ! RMS error for that file
            THEN IF DO$BYPASS(dir_fab)          ! bypassable error?
                 OR .dir_fab[FAB$H_STS] NEQ RMS$_COF
                 THEN LEAVE process             ! Yes
            ELSE SIGNAL(.dir_fab[FAB$H_STS], .dir_fab[FAB$H_STV], dir_fab);

            $TRACE('DIR$DO $SEARCH found input file');

            ! Fill in the XAB's for use only when needed

            IF .list_level NEQ DIU$K_LIST_BRIEF
            THEN BEGIN
                 $OPEN (FAB=dir_fab);
                 !$DISPLAY (FAB=dir_fab);
                 IF NOT $RMS_STATUS_OK(dir_fab)
                 THEN IF DO$BYPASS(dir_fab)
                      THEN LEAVE process        ! Handle bypassable RMS errors
                 ELSE SIGNAL(.dir_fab[FAB$H_STS],! Some other type of RMS error
                             .dir_fab[FAB$H_STV], 
                             dir_fab);
                 $TRACE('DIR$DO XABs filled in OK by $OPEN');

                 $CLOSE(FAB=dir_fab);   ! Close the file now that XABs filled
                 IF NOT $RMS_STATUS_OK(dir_fab)
                 THEN SIGNAL(.dir_fab[FAB$H_STS],
                             .dir_fab[FAB$H_STV], dir_fab);
                 END;

           DIR$BREAK_CHECK();           ! Print a break line if we need to

           ! Display the file attributes at the appropriate level

           CASE .list_level FROM DIU$K_LIST_BRIEF TO DIU$K_LIST_FULL OF
                SET
                [DIU$K_LIST_BRIEF]   : DIR$BRIEF();
                [DIU$K_LIST_NORMAL]  : DIR$NORMAL();
                [DIU$K_LIST_FULL]    : DIR$FULL();
                [INRANGE,OUTRANGE]   : RETURN(DIU$_INV_FUN_CODE);
                TES;

           END;                         ! End of process block

     $SEARCH(FAB=dir_fab);              ! Loop for each file please

     S$BREATHE();                       ! Take a hit of fresh air and sunshine

END;                                    ! End of for all files loop

dir_fab[FAB$V_DRJ] = 0;                 ! Light "don't release JFN" bit

$CLOSE(FAB=dir_fab);                    ! Flush the link

DIR$TOTAL();                            ! Output directory total

END;
%SBTTL 'DIU$DIRECTORY - execute DIRECTORY request'

GLOBAL ROUTINE DIU$DIRECTORY (request : REF $DIU_BLOCK) = 
BEGIN
!++
! FUNCTIONAL DESCRIPTION:
!
!       Take a directory listing.  Search  the comma listed or wildcard  source
!       files given, calling DO$DIR for each file spec given.
!
! FORMAL PARAMETERS:
!
!       request: Address of a DIU request block which contains files to list
!--

LOCAL eob,
      src,
      src_len,
      remaining,   
      next_ptr,
      dest,
      dest_len,
      status,
      show_nam : $NAM_DECL,
      show_fab : $FAB_DECL,
      show_rab : $RAB_DECL,
      show_rsa : VECTOR[CH$ALLOCATION(NAM$K_MAXRSS)],
      current  : $STR_DESCRIPTOR(CLASS=FIXED);

$TRACE(DIU$DIRECTORY);

IF .request[DIU$H_DESTINATION_FILESPEC] EQL 0
THEN SIGNAL(DIU$_BUG);                  ! Punt if no output filespec

! Save the directory listing level, init grand totals

list_level = .request[DIU$Z_LIST_LEVEL];        ! Copy list level from req blk

grand_files = 0;                        ! Grand total files
grand_dirs = 0;                         ! Grand total directories
grand_pages = 0;                        ! Grand total pages (20 only)
grand_blocks = 0;                       ! Grand total blocks (everything else)
nod_len = 0;                            ! Remembered node length
dev_len = 0;                            ! Remembered device length
dir_len = 0;                            ! Remembered directory length

! Get the output file pointed to for today please

dest = CH$FIND_CH(.request[DIU$H_DESTINATION_FILESPEC],
                  CH$PTR(request[DIU$T_DESTINATION_FILESPEC]),
                  $ETG);
dest_len = CH$A_RCHAR(dest);
dest = CH$PLUS(.dest,1);
CH$WCHAR(0,CH$PLUS(.dest,.dest_len));

$TRACE('DIU$DIRECTORY Initializing output RMS structures');

! Init the RMS blocks for the output file, create the output file

$FAB_INIT(FAB=show_fab,  FNA=.dest, FAC=PUT,  RFM=STM,
          FOP=SUP,       NAM=show_nam);
$NAM_INIT(NAM=show_nam,  RSS=NAM$K_MAXRSS,   RSA=CH$PTR(show_rsa));
$CREATE(FAB=show_fab,ERR=RMS$SIGNAL);

! Set up the $RAB for the file and connect it

$TRACE('DIU$DIRECTORY $CREATE went OK, connecting $RAB');
$RAB_INIT(RAB=show_rab,FAB=show_fab,ROP=WBH);
$CONNECT(RAB=show_rab,ERR=RMS$SIGNAL);

fao_rab = show_rab;                     ! Point fao to the show rab
fao_rab[RAB$A_RBF] = fao_buf;           ! Point the RAB to the FAO buffer

$TRACE('DIU$DIRECTORY Output file setup OK');

! Find the first file in the source buffer, if empty send our regrets..

eob = CH$PLUS(CH$PTR(request[DIU$T_SOURCE_FILESPEC]),
              .request[DIU$H_SOURCE_FILESPEC]);
src = CH$FIND_CH(.request[DIU$H_SOURCE_FILESPEC],
                 CH$PTR(request[DIU$T_SOURCE_FILESPEC]),
                 $ETG);
IF CH$FAIL(.src)
THEN SIGNAL(DIU$_INV_STR_LENGTH);

$TRACE('DIU$DIRECTORY Beginning processing of source file list');

DO BEGIN                                ! Start filespec processing loop
   src_len = CH$A_RCHAR(src);           ! Get length of filespec
   src = CH$PLUS(.src,1);               ! Point to the filespec

   $STR_DESC_INIT(DESC=current,         ! make fixed descriptor to that please
                  STRING=(.src_len,.src));

   $TRACE_FAO('DIU$DIRECTORY Calling DIR$DO for !AS', current);

   DIR$DO(current);                     ! Do the work on one source spec

   $TRACE('DIU$DIRECTORY Checking for more files');

   remaining = CH$DIFF(.eob,.src);              ! Get remaining characters
   src = CH$FIND_CH(.remaining,.src,$ETG);      ! Find the next ETG byte
   next_ptr  = CH$PLUS(.src,1);                 ! Point to next file length

END UNTIL CH$RCHAR(.next_ptr) EQL $NUL;         ! Exit if filespec length 0

DIR$GRAND_TOTAL();                      ! Write grand totals

$CLOSE(FAB=show_fab,ERR=RMS$SIGNAL);    ! Close output file

! Tell me what you wrote out and return

$MSG_FAO(' Directory listing written to !AD', 
         .show_nam[NAM$H_RSL],.show_nam[NAM$A_RSA]);

RETURN DIU$_NORMAL; 

END;                                    ! DIU$DIRECTORY
END 
ELUDOM