HPlogo   Communicator 3000 MPE/iX Release 6.0 (Platform Software Release C.60.00):
HP 3000 MPE/iX Computer Systems
> Chapter 9 Year 2000 Enhancements

9.3: New Date Intrinsics for MPE/iX

MPE documents

Complete PDF

 

Table of Contents

Index

 

⇓ Page Bottom

⇑ Page Top

 

Year 2000 Enhancements for MPE/iX

VPLUS Enhancements for Turn-of-Century

by M Gopalakrishnan
Commercial Systems Division

This release of the MPE/iX operating system has enhanced intrinsics for handling date formats. These intrinsics are targeted to support widely used existing date formats in MPE/iX and three new date formats.

The date intrinsics are broadly categorized as follows:
  • HPDATECONVERT — Converting dates from one supported format to another.
  • HPDATEFORMAT — Converting the supported format dates to the display formats desired by the user.
  • HPDATEDIFF — Determining the number of days that separate two given dates.
  • HPDATEOFFSET — Adding/subtracting an offset (days) to/from the given date.
  • HPDATEVALIDATE — Validating the given date for conformance to a supported date format.
  • HPCALENDAR, HPFMTCALENDAR — Using the new 32-bit HPCALENDAR format.
The following sections will discuss various date formats and the syntax and semantics of new date intrinsics. COBOL and Pascal programs using these intrinsics are provided as examples.

Existing Date Formats

Existing date formats that are widely used are summarized in the following table, "Existing Date Formats." The column "Sortable?" indicates whether the dates can be sorted (either numerically or lexicographically depending on how they are stored). The last column "Y2K Ready?" indicates whether dates in the format under consideration can be used to represent dates beyond 1999-12-31. The column "#Bytes" represents the number of bytes required to store the date format.

Table 9-2 Existing Date Formats

Storage Type # Bytes Explanation Sortable? Y2K Ready?
longint [1] 8 Microseconds since 1970-01-01 yes yes
(MPE time-stamp)
integer 4 Upper 2 bytes: year yes yes
next byte: month of year
bottom byte: day of month
integer 4 Upper 2 bytes: year yes yes
bottom 2 bytes: day of year
integer 4 Seconds since 1970-01-01 yes yes
(POSIX.1 time() format; valid through 2038-01-18)
shortint 2 Upper 7 bits: #years since 1900 yes yes
Lower 9 bits: day of the year
(CALENDAR format; valid up to 2027-12-31)
integer 4 YYMMDD date yes no
integer 4 MMDDYY date no no
integer 4 DDMMYY date no no
ASCII [2] 6 YYMMDD date yes no
ASCII 6 MMDDYY date no no
ASCII 6 DDMMYY date no no
ASCII 6 YYMMDD date YY:MM3000 date [3] yes yes
ASCII 6 MMDDYY date YY:MM3000 date no yes
ASCII 6 DDMMYY date YY:MM3000 date no yes

[1] "integer" and "longint" are binary values.

[2] "ASCII" means ASCII character code.

[3] MM3000 dates are represented as in the MM3000 product which used the ASCII letters "A" - "Z" for decades starting with the year 2000.

New Date Formats

Three new date formats are listed in the table, "HP Standard Formats." The first format is an integer representation of the ISO 8601 date format. The second is an ASCII representation of the same. The last date format is an extension of the existing 16-bit CALENDAR date format to a 32-bit format. These date formats will be referred to as "HP Standard Formats."

Table 9-3 HP Standard Formats

Storage Type # Bytes Explanation Sortable? Y2K Ready?
integer 4 YYYYMMDD date yes yes
ASCII 8 YYYYMMDD date yes yes
integer 4 Upper 23 bits: #years since 1900 yes yes
Bottom 9 bits: day of the year.
(Extension of the existing CALENDAR format.)

Supported Date Formats

The "HP Standard Formats" and "Existing Date Formats" are combined in the following table, "Supported Date Formats." Each date format is assigned a date type code. The date intrinsics support these date formats and date type codes.

Table 9-4 Supported Date Formats

Date Type Code Storage Type # Bytes Explanation Sortable? Y2K Ready?
1 longint 8 MPE time-stamp yes yes
(microseconds since 1970-01-01)
2 integer 4 Upper 2 bytes: year yes yes
next byte: month of year
bottom byte: day of month
3 integer 4 Upper 2 bytes: year yes yes
bottom 2 bytes: day of year
4 integer 4 Upper 23 bits: #years since 1900 yes yes
bottom 9 bits: day of the year.
(analogous to the existing CALENDAR format.)
10 integer 4 Seconds since 1970-01-01 yes yes
(POSIX.1 time() format; valid through 2038-01-18)
14 shortint 2 Upper 7 bits: #years since 1900 yes yes
Lower 9 bits: day of the year
(CALENDAR format; valid up to 2027-12-31)
15 integer 4 YYMMDD date yes no
16 integer 4 MMDDYY date no no
17 integer 4 DDMMYY date no no
18 integer 4 YYYYMMDD date yes yes
25 ASCII 6 YYMMDD date yes no
26 ASCII 6 MMDDYY date no no
27 ASCII 6 DDMMYY date no no
35 ASCII 6 [1] YYMMDD date YY:MM3000 date yes yes
36 ASCII 6 MMDDYY date YY:MM3000 date no yes
37 ASCII 6 DDMMYY date YY:MM3000 date no yes
38 ASCII 8 YYYYMMDD date yes yes

[1] MM3000 dates are represented as in the MM3000 product which used the ASCII letters "A" - "Z" for decades starting with the year 2000.

Special Date Values

Special date values to represent UNKNOWN, INVALID, NEVER, NEEDED, EXPIRED, and ILLEGAL dates are defined for dates in "HP standard formats." The following table captures this date value information. The other date formats have been in existence for some time and hence the special date values are not defined for them. These special dates, when passed to the HPDATEFORMAT intrinsic, will result in a corresponding output string. For example, passing "00000103" to HPDATEFORMAT will result in the output string: "NEEDED".

When special dates are passed to HPDATECONVERT, HPDATEOFFSET, and HPDATEDIFF intrinsics as input dates, an error status is returned and the output date is initialized to a binary zero value or a blank string, depending on the date type of the output date.

Table 9-5 Special Date Values

DataType Format Un-
known
Invalid Never Needed Expired Illegal
4 YYYYDDD 0 0000367 0000368 0000369 0000370 0000371
18 YYYYMMDD 0 00000101 00000102 00000103 00000104 0000105
38 YYYYMMDD "blank" "0000010" "00000102" "00000103" "00000104" "0000105"

New Date Intrinsics

The date intrinsics support dates in the range 0001-01-01 through 9999-12-31. They use the Gregorian calendar for all calculations beyond 1753. The calendar followed by the intrinsics ignores the fact that calendars in different countries changed at different times and some dates were dropped from the calendar (around the year 1753). The validity of dates prior to 1753 cannot be guaranteed. All intrinsics accept byte-aligned input/output date parameters, and all these intrinsics are NM callable.

On an error, the intrinsics initialize the output parameters to either a binary zero or a blank string depending on the type of the parameter.

Though the date type "4" can represent years beyond 9999, a year beyond 9999 (which needs five digits/characters) is considered an error.

It is recommended that character array based output date parameters are initialized to blanks before passing them as arguments to these intrinsics.

New errors and warnings are documented in the System Message Catalog, SYSCAT.PUB.SYS, under the date intrinsics subsystem number 529.

HPDATECONVERT

This intrinsic converts the dates from one supported format to another.

Syntax

                 I32V      *         I32V       *          I32    I32V
 HPDATECONVERT ( inputcode,inputdate,outputcode,outputdate,status,cutoff )
  
Parameters
inputcode

is a 32-bit signed integer by value.

The value should be one of the date type codes listed in the table, "Supported Date Formats."

inputdate

varies for type by reference.

The interpretation depends upon the value of inputcode. See the table, "Supported Date Formats," for the supported date codes and their layouts.

outputcode

is a 32-bit signed integer by value.

The value should be one of the date type codes listed in the table, "Supported Date Formats."

outputdate

returns the date as per the format chosen by the outputcode parameter.

See the table, "Supported Date Formats," for the supported datecodes and their layouts.

status

is the HPE_STATUS parameter through which the error codes are returned. A value of 0 indicates no error and no warnings.

cutoff

is a 32-bit signed integer by value (optional).

This is used in validating the input parameter when the input date has two-digit year. This is a required parameter for dates with two-digit years. In all other cases, this parameter is ignored.

If the cutoff parameter is given as -1, the value of the CI environment variable HPSPLITYEAR is used as the cutoff year.

This parameter's value should be in the range 0 .. 100. If the value of the parameter is 50, two digit years in the range 0 .. 49 will translate to 200 .. 2049 and those in the range 50 .. 99 will be translated to 1950..1999. If you specify the cutoff year as 70, the mapping will be 0 .. 69 as 2000 .. 2069 and 70 .. 99 as 1970 .. 1999.

HPDATEFORMAT

You can use this routine to format the dates that can be combinations of display formats as explained below. Many of these elements are taken from ALLBASE/SQL date formats.

You can convert dates in the "Supported Date Formats" to a display string of your choice (with restrictions). The HPDATEFORMAT intrinsic will accept these format strings. The format specification strings can have the following syntax:

Syntax

 [{FormatElement}{Punctuation}]
  
Table 9-6 Valid Parameters for FormatElement

Format Element Punctuation
CC Century (01 to 99)
YYYY Year (0001 to 9999)
YY Year of century (00 to 99) with leading zeros suppressed (0 to 99)
Q Calendar quarter of the year (1 to 4)
MM Month of the year (01 to 12)
ZMM Month of the year with leading zeros suppressed (1 to 12)
DD Day of the month (01 to 31)
ZDD Day of the month with leading zero suppressed (1 to 31)
DDD Day of the year (001 to 366).
ZDDD DDD with leading zeros suppressed (1 to 366)
D Day of the week (1 to 7 where Sunday is 1, Monday is 2,...)
WW Week of the year (01 to 53)
ZWW Week of the year with leading zero suppressed (1 to 53)
Mon Month of the year in ASCII format (Jan, Feb,...)
Day Day of the week in ASCII format (Sun, Mon,...)
MON Month of the year in ASCII (uppercase) format (JAN, FEB,...)
DAY Day of the week in ASCII (uppercase) format (SUN, MON,...)

Table 9-7 Valid Characters for Punctuation

Char Description
- Hyphen
/ Slash
. Dot
Blank
, Comma
Null (or no delimiter)

Thus, YYYY.MON.DAY, YY/MM/DD, DDMONYY, and DD-ZMM-YYYY are valid date formats. For example, "31 Jan 1997" when formatted through DD-ZMM-YYYY results in "31-1-1997," formatted through YYYY.MON.DAY results in "1997.JAN.FRI," while YYYYMMDD results in "19970131."

Note
NOTE: Mixing the NULL punctuation character with other punctuation characters is not allowed. Thus, YYYY/MM/DD is a valid format, while YYYYMM/DD is not. fmtdatelen parameter is length of formatspec parameter including the null termination character.

Syntax

                I32V     *         CA         CA      I32        I32    I32V
 HPDATEFORMAT ( datecode,inputdate,formatspec,fmtdate,fmtdatelen,status,cutoff ) 
  
Parameters
datecode

is a 32-bit signed integer by value.

This value should be one of the date type codes listed in the table, "Supported Date Formats."

inputdate

is the input date.

The interpretation depends upon the value of datecode. See the table, "Supported Date Formats," for the supported datecodes and their layouts.

formatspec

is a character array (required).

This should be a NULL terminated string as per the syntax explained above in the table, "Format Specification Strings."

fmtdate

is a character array (required).

This array size should be at least that of formatspec. On return, it will contain the date formatted as per the formatspec.

If an invalid date is passed, on return from the intrinsic its contents will be UNKNOWN. For the "HP Standard Formats," if special values in the table, "Special Date Values," are passed for date parameter, on return from the intrinsic, the value of the string will be appropriately initialized. For example, for the date type 18, the initialized values for different special date values are as follows:

Date Value Returned Contents
00000000 "UNKNOWN"
00000101 "INVALID"
00000102 "NEVER"
00000103 "NEEDED"
00000104 "EXPIRED"
00000105 "ILLEGAL"

If the character array passed does not have enough space to hold the special values or the formatted date, the behavior is undefined.

fmtdatelen

is a 32 bit integer by reference (required).

On input, it is the length of the formatspec parameter.

On return, it represents the number of characters HPDATEFORMAT placed into fmtdate.

status

is the HPE_STATUS parameter through which the error codes are returned. A value of 0 indicates no error and no warnings.

cutoff

is a 32-bit signed integer by value (optional).

This is used in validating and converting from two-digit to four-digit years. (See HPDATECONVERT documentation for more information on the cutoff parameter).

HPDATEDIFF

This intrinsic determines the number of days that separate two given dates.

Syntax


              I32V     *         *          I32        I32    I32V
 HPDATEDIFF ( datecode,firstdate,seconddate,diffindays,status,cutoff )
  
Parameters
datecode

is a 32-bit signed integer by value.

This value should be one of the date type codes listed in the table, "Supported Date Formats."

firstdate

is the first input date. The interpretation depends upon the value of datecode. See the table, "Supported Date Formats," for the supported datecodes and their layouts.

seconddate

is the second input date. The interpretation depends upon the value of datecode. See the table, "Supported Date Formats," for the supported datecodes and their layouts.

diffindays

is the number of days difference between the two dates, computed as: seconddate minus firstdate. Thus, if seconddate is earlier than firstdate, diffindays will be negative.

status

is the HPE_STATUS parameter through which the error codes are returned. A value of 0 indicates no error and no warnings.

cutoff

is a 32-bit signed integer by value (optional).

This is used in validating and converting the two-digit years to four-digit years before computing the difference. (See HPDATECONVERT documentation for more information on cutoff parameter.)

HPDATEOFFSET

This intrinsic adds or subtracts a specified offset to or from the given date.

Syntax


                I32V     *         I32V   *          I32    I32V
 HPDATEOFFSET ( datecode,inputdate,offset,outputdate,status,cutoff )
  
Parameters
datecode

is a 32-bit signed integer by value

This value should be one of the date type codes listed in the table, "Supported Date Formats."

inputdate

is the input date. The interpretation depends upon the value of datecode. See the table, "Supported Date Formats," for the supported datecodes and their layouts.

offset

is a 32-bit signed integer by value.

The number of days to be added to the input date. A negative value will result in a subtraction.

outputdate

is the output date. The result of the date offset operation. The interpretation depends upon the value of datecode. See the table, "Supported Date Formats," for the supported datecodes and their layouts.

status

is the HPE_STATUS parameter through which the error codes are returned. A value of 0 indicates no error and no warnings.

cutoff

is a 32-bit signed integer by value (optional).

This is used in validating and converting the two-digit years to four digit ones before computing the difference. (See HPDATECONVERT documentation for more information.)

HPDATEVALIDATE

This intrinsic checks the validity of the given date with respect to the supported formats given in the table, "Supported Date Formats."

Syntax

 I32                        I32V     *         I32V
 result := HPDATEVALIDATE ( datecode,inputdate,cutoff )
  
Parameters
datecode

is a 32-bit signed integer by value.

This value should be one of the date type codes listed in the table, "Supported Date Formats."

inputdate

is the input date.

The interpretation depends upon the value of datecode. See the table, "Supported Date Formats," for the supported datecodes and their layouts.

cutoff

is a 32-bit signed integer by value (optional).

This is used in validating dates with two digit years. (See HPDATECONVERT documentation for more information on cutoff parameter).

result

is a 32-bit signed integer (assigned functional return).

This value will be 0 if the inputdate conforms to the date format represented by datecode. If it is not so, its value will be positive. If an error has occurred in evaluating the conformance, its value will be negative. This return value ranges from -999 to 1.

HPCALENDAR

The new HPCALENDAR intrinsic returns the date in the supported date type code 4 listed in the table, "Supported Date Formats." (This is also a HP standard format.)

Syntax

 I32
 date := HPCALENDAR ;
    
where date is the 32-bit unsigned integer (assigned functional return).

This returns the calendar date in the following format:

Table 9-8 HPCALENDAR Date Format

Bits Value/Meaning
23:9 Day of year
0:23 Year since 1900

HPFMTCALENDAR

This a new routine to handle HPCALENDAR format. It does the same job as FMTCALENDAR except that it accepts the 32-bit integer returned by HPCALENDAR intrinsic.

Syntax

                 I32V  CA
 HPFMTCALENDAR ( date, formatdate )
  
Parameters
date

is a 32-bit signed integer by value

This holds the calendar date, in the same format as the HPCALENDAR intrinsic (that is, date type 4).

formatdate

returns the formatted calendar date in a 17-character array. If the day of the month is less than 10, a blank precedes it. For example,

 FRI, JAN  6, 1989
        

Examples

Following are two examples:
  • The Pascal program example uses most of the new date intrinsics.
  • The COBOL program example uses the HPDATECONVERT intrinsic.
Pascal Example of New Date Intrinsics

 $standard_level 'hp_modcal'$

 Program dateintr(input,output);

      {Constants for the different date types.}
 const
       hp_dt_mpe_time_stamp_fmt   = 1;
       hp_dt_packed_yymmdd_fmt    = 2;
       hp_dt_packed_yyddd_fmt     = 3;
       hp_dt_new_calendar_fmt     = 4;

       hp_dt_posix_time_stamp_fmt = 10;

       hp_dt_calendar_fmt         = 14;
       hp_dt_int_yymmdd_fmt       = 15;
       hp_dt_int_mmddyy_fmt       = 16;
       hp_dt_int_ddmmyy_fmt       = 17;
       hp_dt_int_yyyymmdd_fmt     = 18;

       hp_dt_ascii_yymmdd_fmt     = 25;
       hp_dt_ascii_mmddyy_fmt     = 26;
       hp_dt_ascii_ddmmyy_fmt     = 27;

       hp_dt_mm3000_yymmdd_fmt    = 35;
       hp_dt_mm3000_mmddyy_fmt    = 36;
       hp_dt_mm3000_ddmmyy_fmt    = 37;
       hp_dt_ascii_yyyymmdd_fmt   = 38;
 
       {Type definitions for the program}
 type
     pac_20       = packed array [1..20] of char;
     iptr_type    = ^integer;

 VAR
     date2_pac        : pac_20;      {Dates in packed character arrays.}
     fmt_pac          : pac_20;
     print_pac        : pac_20;

     cutoff           : integer;     {To represent cutoff date.}
     date1            : integer;     {Working dates.}
     date1_18         : integer;
     date2_18         : integer;
     date3_18         : integer;
     i                : integer;     {temp variables.}
     inptype          : integer;
     j                : integer;
     outtype          : integer;
     print_len        : integer;     {Length of formatted date.}
     temp_date        : integer;     {Another temp variable}

     status           : integer;

 function HPCALENDAR : integer; intrinsic;
 procedure HPFMTCALENDAR; intrinsic;
 function HPDATEVALIDATE : integer; intrinsic;
 procedure HPDATEFORMAT ; intrinsic;
 procedure HPDATECONVERT ; intrinsic;
 procedure HPDATEOFFSET ; intrinsic;
 procedure HPDATEDIFF ; intrinsic;

 begin
    {Initialize the variables.}
    cutoff  := 50;
    inptype := hp_dt_int_ddmmyy_fmt;

    date2_pac := '960121'; {The YYMMDD date in Supported format '25'.}
    date1     := 230196;   {The DDMMYY date in Supported format '17'.}


         {conversion from one non standard formatto another}
    HPDATECONVERT(inptype,date1,hp_dt_ascii_yymmdd_fmt,fmt_pac,
              status,cutoff);
    if( status <> 0) then
        writeln('HPDATECONVERT intrinsic returned error',status);

    {Convert 'date1' to the YYYYMMDD integer format. With 50}
    {as the cutoff year.                                    }
    outtype := hp_dt_int_yyyymmdd_fmt;
    HPDATECONVERT(inptype,date1,outtype,date1_18,status,cutoff);
    if( status <> 0) then
        writeln('HPDATECONVERT intrinsic returned error',status);

    {Convert the 'date2_pac' to a YYMMDD integer format.}
    {Default cutoff is assumed (i.e., 50)               }
    HPDATECONVERT(hp_dt_ascii_yymmdd_fmt,date2_pac,
        outtype,date2_18,status);
    if( status <> 0) then
        writeln('HPDATECONVERT intrinsic returned error',status);

    {compute the difference between the two dates}
    HPDATEDIFF(outtype, date1_18,date2_18,temp_date,status,cutoff);

    if( temp_date <  0) then
        writeln(date1,' is #',temp_date,
      'days later compared to ',date2_pac)
    else if (temp_date > 0) then
        writeln(date2_pac,' is #',temp_date,
      'days later compared to ',date1)
    else writeln(date1,' is same as ',date2_pac);


       {offset date1_18 by the difference to get the     }
       {value of date2_18 through HPDATEOFFSET intrinsic.}
    HPDATEOFFSET(outtype,date1_18,temp_date,date3_18,status,cutoff);
    if( (date3_18 <> date2_18 ) and (status = 0)) then
        writeln('Error Malfunction of HPDATEOFFSET intrinsic');

       {Use the flexibility in converting the date to a display }
       {string. Use '#0' to NULL terminate the format           }
       {specification string.                                   }
    fmt_pac := 'YY.ZMM.ZDD'#0;
    print_len := 11; { 11 characters including #0 in fmt_pac }
    HPDATEFORMAT(outtype,date1_18,fmt_pac,print_pac,print_len,status);

    if( status <> 0) then
        writeln('HPDATEFORMAT intrinsic returned error',status);

    writeln('The converted date is: ',print_pac);

       {The following demonstrates how a byte array can be      }
       {passed in place of an integer. We are storing the       }
       {integer 230196 in a packed array of characters, which is}
       {an equivalent of a byte array. We subsequently cast the }
       {address of this array to an integer pointer to read the }
       {integer value out of it.                                }
    HPDATECONVERT(hp_dt_int_yyyymmdd_fmt,date1_18,
        hp_dt_int_ddmmyy_fmt,fmt_pac,status);

    $push, type_coercion 'storage'$
    if( iptr_type (addr(fmt_pac))^ = date1 ) then
            writeln( 'Program worked fine! ')
    else    writeln( 'Error, in using byte array parameters !');
    $pop$

 end.
  
The expected output from the program is:

 230196 is #     2days later compared to 960121
 The converted date is : 96.1.23
 Program worked fine!
  
COBOL Example of HPDATECONVERT

       IDENTIFICATION DIVISION.
       PROGRAM-ID.   DATEINTR.
       REMARKS. SAMPLE PROGRAM USING DATE INTRINSICS
       DATA DIVISION.
       WORKING-STORAGE SECTION.
      *Date code for YYMMDD is 25, for YYYYMMDD is 38
       01 INDATE-CODE                  PIC S9(9) VALUE 25.
       01 OUTDATE-CODE                 PIC S9(9) VALUE 38.
       01 SPLITYEAR                    PIC S9(9) VALUE 70.
       01 OUTDATE-YYYYMMDD             PIC X(8) VALUE SPACES.
       01 INDATE-YYMMDD                PIC X(6) VALUE SPACES.
       01 STATUS-VAR.
          05 S-INFO                    PIC S9(4) COMP VALUE 0.
          05 S-SUBSYS                  PIC S9(4) COMP VALUE 0.

       PROCEDURE DIVISION.
       CONVERT-DATE-PARA.
          DISPLAY 'Enter the date in YYMMDD format:'
          ACCEPT INDATE-YYMMDD
          CALL INTRINSIC "HPDATECONVERT" USING
                                         INDATE-CODE
                                         INDATE-YYMMDD
                                         OUTDATE-CODE
                                         OUTDATE-YYYYMMDD
                                         STATUS-VAR
                                         SPLITYEAR
          IF S-INFO NOT = 0
             PERFORM DISPLAY-ERROR
          ELSE
             DISPLAY "Convert Date in YYYYMMDD = " OUTDATE-YYYYMMDD
          END-IF
          STOP RUN.

       DISPLAY-ERROR.
          DISPLAY "HPDATECONVERT FAILED. ERROR = " S-INFO.
  



Year 2000 Enhancements for MPE/iX

VPLUS Enhancements for Turn-of-Century