|Example 5-3 READ_VERIFY.MAR Terminal Driver Programming Example |
.TITLE READ_VERIFY - Read Verify Coding Example .IDENT 'V05-000' .SBTTL DECLARATIONS .DISABLE GLOBAL;; Declare the external system routines and MACRO libraries.; .EXTERNAL LIB$GET_EF .EXTERNAL SCR$ERASE_PAGE .LIBRARY 'SYS$LIBRARY:LIB.MLB' .LIBRARY 'SYS$LIBRARY:STARLET.MLB';; Include files:; $IODEF $TRMDEF;; Macros:;.MACRO ITEM LEN=0,CODE,VALUE .WORD LEN .WORD TRM$_'CODE' .LONG VALUE .LONG 0.ENDM ITEM;; Equated symbols:;INBUF_LEN = 20ESC = ^X1B;; Own storage:;; Build item lists for the read verify QIO;;; Right-justified field;R_ITEM_LIST: ITEM CODE = MODIFIERS, - VALUE = TRM$M_TM_R_JUST ; Right justify ITEM CODE = EDITMODE, - VALUE = TRM$K_EM_RDVERIFY ; Enable read verify ITEM CODE = PROMPT, - VALUE = R_PROMPT_ADDR, - LEN = R_PROMPT_LEN ; Set up prompt ITEM CODE = INISTRNG, - VALUE = R_INISTR_ADDR, - LEN = R_INISTR_LEN ; Set up initial string ITEM CODE = INIOFFSET, - VALUE = R_INISTR_LEN ITEM CODE = PICSTRNG, - VALUE = R_PICSTR_ADDR, - LEN = R_PICSTR_LEN ; Set up picture string ITEM CODE = FILLCHR, - VALUE = <^A/* /> ; clear = *, fill = spaceR_ITEM_LIST_LEN = .-R_ITEM_LISTR_PROMPT_ADDR: .ASCII <ESC> /[12;12H$/R_PROMPT_LEN = .-R_PROMPT_ADDRR_INISTR_ADDR: .ASCII / , /R_INISTR_LEN = .-R_INISTR_ADDRMASK = TRM$M_CV_NUMERIC!TRM$M_CV_NUMPUNCR_PICSTR_ADDR: .BYTE MASK .BYTE MASK .BYTE MASK .BYTE 0 ; Marker character .BYTE MASK .BYTE MASK .BYTE MASKR_PICSTR_LEN = .-R_PICSTR_ADDR;; Left-justified field;L_ITEM_LIST: ITEM CODE = MODIFIERS, - VALUE = TRM$M_TM_CVTLOW!TRM$M_TM_AUTO_TAB ; Upcase input and ; complete on field full ITEM CODE = EDITMODE, - VALUE = TRM$K_EM_RDVERIFY ; Enable read verify ITEM CODE = PROMPT, - VALUE = L_PROMPT_ADDR, - LEN = L_PROMPT_LEN ; Set up prompt ITEM CODE = INISTRNG, - VALUE = L_INISTR_ADDR, - LEN = L_INISTR_LEN ; Set up initial string ITEM CODE = INIOFFSET, - VALUE = 0 ITEM CODE = PICSTRNG, - VALUE = L_PICSTR_ADDR, - LEN = L_PICSTR_LEN ; Set up picture string ITEM CODE = FILLCHR, - VALUE = <^A/* /> ; clear = *, fill = spaceL_ITEM_LIST_LEN = .-L_ITEM_LISTL_PROMPT_ADDR: .ASCII <ESC>/[13;12H Enter Date: /L_PROMPT_LEN = .-L_PROMPT_ADDRL_INISTR_ADDR: .ASCII / - - /L_INISTR_LEN = .-L_INISTR_ADDRMASK1 = TRM$M_CV_NUMERICMASK2 = TRM$M_CV_UPPER!TRM$M_CV_LOWERL_PICSTR_ADDR: .BYTE MASK1 .BYTE MASK1 .BYTE 0 ; Marker character .BYTE MASK2 .BYTE MASK2 .BYTE MASK2 .BYTE 0 ; marker character .BYTE MASK1 .BYTE MASK1L_PICSTR_LEN = .-L_PICSTR_ADDRIN_IOSB: .BLKL 2TT_CHAN: .BLKW 1INBUF: .BLKB INBUF_LENSYSINPUT: .ASCID /SYS$INPUT/SYNC_EFN: .BLKL 1 .PAGE .ENTRY READ_VERIFY ^M < >;; Get the required event flags.; PUSHAL SYNC_EFN CALLS # 1, G^ LIB$GET_EF BLBC R0, ERROR ; Error - branch;; Assign the channel to SYS$INPUT; $ASSIGN_S - CHAN = TT_CHAN - DEVNAM = SYSINPUT ; SYS$INPUT BLBC R0, ERROR ; Branch on error;; Clear the screen; CLRQ -(SP) CALLS #2, G^ SCR$ERASE_PAGE BLBC R0, ERROR;; Do the right-justified read operation; PUSHL #R_ITEM_LIST_LEN PUSHAB R_ITEM_LIST CALLS #2, DO_READ BLBC R0, ERROR;; Do the left-justified read operation; PUSHL #L_ITEM_LIST_LEN PUSHAB L_ITEM_LIST CALLS #2, DO_READ BLBC R0, ERRORERROR: RET .PAGE;++;; DO_READ - do the actual QIO;; Inputs:;; 4(AP) the address of the itemlist; 8(AP) the length of the itemlist;;-- .ENTRY DO_READ, ^M<> $QIOW_S EFN=SYNC_EFN, - CHAN = TT_CHAN, - FUNC = #<IO$_READVBLK!IO$M_EXTEND>, - IOSB = IN_IOSB, - p1 = inbuf, - p2 = #inbuf_len, - p5 = 4(AP), - P6 = 8(AP) BLBC R0, 10$ ; QIO error - branch MOVZWL IN_IOSB, R0 ; Get the terminal driver status. BLBC R0, 10$ ; Terminal driver error - branch; Handle the input...10$: RET .END READ_VERIFY
This chapter describes the use of the pseudoterminal driver (FTDRIVER) and the pseudoterminal software.
A pseudoterminal is a software device that appears as a real terminalto an application communicating with it, but does not require theexistence of a physical terminal. A pseudoterminal consists of twocomponents: the pseudoterminal device and a control program. Thecontrol program acts like a keyboard; that is, anything written to thecontrol program appears on the pseudoterminal device as if thekeystrokes had been typed in at a physical terminal. The controlprogram also acts like a viewport to the pseudoterminal device; thatis, the control program reads anything that is written by the system tothe pseudoterminal device.
A pseudoterminal allows an application to be set up on the control side of the link to communicate with another application that is on the pseudoterminal side. This arrangement allows development of applications that either simulate users or monitor the communication between a real user (at a physical terminal) and an application. As with other devices, the work of the pseudoterminal is performed by a device driver and is tightly coupled to the operating system.
The pseudoterminal driver software includes a set of control connection routines. Applications can use these routines to perform pseudoterminal operations and functions. Appendix D provides the calling conventions for these routines.
6.1 Pseudoterminal Operations
This section contains information on the following pseudoterminaloperations:
- Creating a pseudoterminal
- Canceling a request
- Deleting a pseudoterminal
6.1.1 Creating a Pseudoterminal
To create a pseudoterminal, use the PTD$CREATE routine described in Appendix D. When a pseudoterminal is created, it inherits the current system terminal default attributes unless you specify an alternate set of characteristics. In either case, you cannot use PTD$CREATE to alter the following startup attributes:
- TT$M_CRFILL is cleared. To change this attribute, issue the SET MODE $QIO function.
- TT$M_LFFILL is cleared. To change this attribute, issue the SET MODE $QIO function.
- TT$M_MODEM is cleared. This attribute cannot be changed.
- TT$M_REMOTE is cleared. This attribute cannot be changed.
- TT$M_HOSTSYNC is set. To change this attribute, issue the SET MODE $QIO function.
- TT$M_TTSYNC is set. To change this attribute, issue the SET MODE $QIO function.
- TT2$M_DMA is cleared. To change this attribute, issue the SET MODE $QIO function. Changing it does not alter the behavior of TTDRIVER or the pseudoterminal.
- TT2$M_AUTOBAUD is cleared. To change this attribute, issue the SET MODE $QIO function. Changing it does not alter the behavior of TTDRIVER or the pseudoterminal.
- TT2$M_FALLBACK is cleared. To change this attribute, issue the SET MODE $QIO function.
- TT2$M_HANGUP is cleared. To change this attribute, issue the SET MODE $QIO function.
- TT2$M_DCL_MAILBX is cleared. This attribute cannot be changed.
When you create a pseudoterminal, you can specify a repeatingasynchronous system trap (AST) to be delivered when the terminalconnection is freed. This AST can be supplied only when thepseudoterminal is created, and it cannot be deleted. A terminal isfreed when a process logs out or deassigns the last channel to thedevice. The AST allows the control program to determine whether or nota user of a pseudoterminal is using it. At this point, the controlprogram can reuse or delete the pseudoterminal by deassigning thecontrol channel.
6.1.2 Canceling a Request
To cancel a queued control connection request, the control program usesthe PTD$CANCEL routine. This routine enables the pseudoterminal driverto differentiate between control requests and terminal requests thatare being canceled. This routine cannot be used to flush eventnotification ASTs.
6.1.3 Deleting a Pseudoterminal
To delete the pseudoterminal, the control program uses the PTD$DELETEroutine. When a pseudoterminal is deleted, any process that is usingthe pseudoterminal (except the control process) is disconnected. If youhave the TT2$M_DISCONNECT bit set in the default terminalcharacteristics parameter (TTY_DEFCHAR2) and virtual terminals havebeen enabled (see Section 220.127.116.11), you get a virtual terminal uponlogging in to a pseudoterminal. In this case, the process is not loggedout, but the virtual terminal is disconnected from the pseudoterminal.
The PTD$DELETE request causes any pending I/O for the control programto be aborted. It deletes any queued event notification ASTs andreturns the I/O buffers to the application. It also causes thepseudoterminal unit control block (UCB) to be deleted once thereference count returns to zero.
If an application exits without calling PTD$DELETE, the pseudoterminalis still deleted.
6.2 Pseudoterminal Driver Features
The terminal portion of a pseudoterminal is similar to a regularterminal. The pseudoterminal driver provides the following features:
- Type-ahead buffer
- Specifiable or default line terminators
- Special operating modes, such as NOECHO and PASTHRU
- Escape sequence detection
- Terminal/mailbox interaction
- Terminal control characters, such as Ctrl/S and Ctrl/Q for starting and stopping output, Ctrl/O for discarding output, and all other special characters that are handled by the standard terminal driver
- Limited full-duplex operation (simultaneously active read and write requests)
For more information on these features, see Section 5.2.
6.3 Pseudoterminal Driver Device Information
The pseudoterminal inherits its device characteristics from the systemdefault parameters, with the following exceptions:
- The device inherits initial device characteristics from the SYSGEN-supplied default values. You can modify the device characteristics during device creation by supplying new characteristics.
- The HOSTSYNC terminal characteristic is always set.
- The device is set to NOMODEM and cannot be set to MODEM.
- The device is set not to time output character transmission. Hardware controllers time output character transmission to determine whether the controller is broken.
You can obtain information on pseudoterminal characteristics by usingthe Get Device/Volume Information ($GETDVI) system service, asdescribed in Section 5.3 and the OpenVMS System Services Reference Manual.
Applications should assign a channel other than the control channel inorder to read data from, write data to, read, or alter thepseudoterminal characteristics. An attempt to perform such I/O with thecontrol channel, or any other attempt to queue an illegal or unsafe I/Orequest, results in an SS$_CHANINTLK error.
6.4 I/O Buffers
When you create a pseudoterminal, you must provide at least one page to be used as an I/O buffer.
On Alpha systems, you can allocate one page and divide it into I/O buffers as needed.
On VAX systems, each page becomes one I/O buffer. You should allocate no more than six I/O buffers for each pseudoterminal.
No read or write request should reference more than one I/O buffer at a time. The I/O buffers must be page aligned; therefore, you should create these pages with the $EXPREG system service or the LIB$GET_VM_PAGE routine. The pages are owned by the driver until you delete the pseudoterminal. The application is responsible for managing the pages and cannot use buffers that are owned by another pseudoterminal. The application must decide whether to delete the buffers when they are freed by the driver or to reuse them.
The I/O buffers must be valid pages in virtual address space. Creating or deleting an I/O buffer does not alter the contents of the pages.
The low-order word of the status information longword contains the status of the request. The high-order word of the status information longword contains the actual number of bytes that are read or written.
Assume that an I/O buffer starting at 200 hexadecimal is available foruse. If you want to read 20 bytes from the pseudoterminal, thereadbuf address would be 200, and thereadbuf_len would be 20. An application can use therest of this buffer for other purposes, including reading or writing tothe pseudoterminal. Figure 6-1 shows how the buffer would look.
Figure 6-1 Buffer Layout
6.5 Pseudoterminal Functions
This section discusses the following pseudoterminal functions:
- Reading data
- Writing data
- Using write with echo
- Flow control
- Event notification
6.5.1 Reading Data
To read data from the pseudoterminal, the control program uses thePTD$READ routine. The read request completes with a minimum of onecharacter and a maximum of the number of characters requested. The readoperation completes when the pseudoterminal has characters to output.If a read request is issued and no data is available, the read requestis queued and then completed at a later time.
An application that issues an asynchronous pseudoterminal read can usethe $SYNCH system service to find out when the read completed. Theefn argument for the $SYNCH service must be the sameas the efn specified in the original PTD$READ call,and the iosb for the $SYNCH service must match thereadbuf of the PTD$READ call.
6.5.2 Writing Data
To write data to the pseudoterminal, the control program uses thePTD$WRITE routine. The write request allows you to specify a buffer toreceive any output that the write request generates; you do not need toissue a separate read request to read this data. When you use an echobuffer, the control application can significantly reduce the number ofI/O requests required.
An application can issue only one write request at a time. Once thewrite request completes, the application must check the write bufferstatus longword to see whether all the data supplied was written. Ifnot, the application must issue additional write requests until all thedata has been accepted.
6.5.3 Using Write with Echo
If a read request is pending when a write-with-echo request is issued,the echo data is placed in the echo buffer. If more data is echoed thancan fit in the echo buffer, the remaining data is placed in the pendingread requests buffer. If no pending read exists, the data is held bythe driver until another request that can take the data is issued. Boththe read and the write with echo must use completion ASTs to allow thedriver to report request completions to the application in the correctorder.
If an application is not using the write-with-echo capability, the application should avoid using completion ASTs if possible. Unnecessary use of completion ASTs significantly increases the number of instructions needed to complete a read or write operation.
When using write with echo, both the wrtbuf andechobuf arguments contain I/O status information. Anapplication must check both of these status longwords if the PTD$WRITEcompletes successfully. If a write operation wrote no characters,characters might still be in the echo buffer. If no data was echoed,the status in the echobuf is SS$_NORMAL with zerobytes transferred.
6.5.4 Flow Control
By default, the driver attempts to notify the control program of dataoverrun or loss. The pseudoterminal sends an XOFF AST when thetype-ahead buffer is getting full. Once the pseudoterminal delivers anXOFF AST, the pseudoterminal also returns a status of SS$_DATAOVERUNwith the actual number of characters input. This prevents a singlerequest from flooding the type-ahead buffer. If a control program makesrepeated attempts to insert data after receiving the SS$_DATAOVERUNmessage, it can flood the terminal type-ahead buffer. When thetype-ahead buffer has filled, the pseudoterminal returns the status ofSS$_DATALOST.
If the control program is writing to the terminal or terminal driver,it should let the terminal and terminal driver handle flow control. Todo this, the application should enable all three input flow controlnotification ASTs. The control program should write a DC1 to theterminal if an XON AST is delivered. It should write a DC3 to aterminal if an XOFF AST is delivered, and write a BELL character to theterminal if the BELL AST is delivered. These signals allow the terminalto decide what to do with the flow control data. The application shouldignore the SS$_DATAOVERUN and SS$_DATALOST return status and continuewriting data to the pseudoterminal.
6.5.5 Event Notification
This section describes how the pseudoterminal driver providesnotification of important driver events.
18.104.22.168 Input Flow Control
The driver provides three ways to indicate when the class driver wantsto stop input and one way to signal when it is safe to resume output:
- The driver returns a status of SS$_DATAOVERUN and the number of characters input for the control program write.
- The control program can enable a BELL attention AST to be delivered when the class driver calls the PTD$SET_TERMINAL_NOTIFICATION routine. This AST is delivered if the pseudoterminal does not have the HOSTSYNC attribute set. If only a BELL or only an XOFF AST event is enabled and an XOFF or a BELL AST needs to be delivered, the AST that is available is delivered.
- The control program can enable an XOFF attention AST to be delivered when the class driver calls the PTD$SET_TERMINAL_NOTIFICATION routine. This AST is delivered if the pseudoterminal has the HOSTSYNC attribute set.
- The control program can enable an XON attention AST to be delivered when the class driver calls the PTD$SET_TERMINAL_NOTIFICATION routine. This AST is delivered only if the pseudoterminal has the HOSTSYNC attribute set.
22.214.171.124 Output Stop
The Output Stop AST tells the control program that the terminal driveris stopping output. This keeps the control program from having todetermine whether an XOFF written to the control side is being treatedby the terminal driver as flow control or data.
126.96.36.199 Output Resume
The Output Resume AST tells the control program that the terminaldriver wants to resume output. This AST can be delivered at any time,even if output is active or has previously been stopped. The controlprogram should always restart output processing when it receives thisAST.
188.8.131.52 Characteristics Changed
The Characteristics Changed AST tells the control program that theterminal driver has called the pseudoterminal CHANGE CHARACTERISTICSroutine. This routine is called whenever the terminal driver haschanged the device characteristics. The control program should thenread the pseudoterminal characteristics to determine what has changed.
184.108.40.206 Output Abort
The Output Abort AST tells the control program that the terminal driverhas called the pseudoterminal ABORT OUTPUT routine. This routine iscalled when the terminal driver wants to flush any outstanding outputdata. The control program should flush any internally buffered datawhen this AST is received.
220.127.116.11 Terminal Driver Read Events
Three special event types notify the control program when a terminalread request starts and finishes. By default, the pseudoterminal doesnot deliver the read notification ASTs associated with these events.The PTD$SET_EVENT_NOTIFICATION routine must be used explicitly toenable or disable their delivery.
- Start Read---Tells the control program that the terminal driver is starting a read request. Some applications require this in order to know when to start inputting a logged session script.
- Middle Read---Tells the control program that the terminal driver has finished writing the prompt string if one was supplied.
- End Read---Tells the control program that the terminal driver has finished a read request.
Once an event notification AST is enabled, it continues to be delivereduntil it is canceled, or until the device is deleted. Thischaracteristic allows the control program to enable the AST once, whichgreatly reduces the risk of missing multiple rapid occurrences of anevent. If the driver cannot get sufficient resources to deliver thenotification AST, that report is lost. Only one AST per event isallowed, and attempts to specify multiple ASTs result in use of thelast one specified.
To enable or disable event notification, the control program uses thePTD$SET_EVENT_NOTIFICATION routine, which is described in Appendix D.
6.6 Pseudoterminal Driver Programming Example
Example 6-1 shows how to use the pseudoterminal. (The example is alsoincluded in the SYS$EXAMPLES directory.) This section begins with abrief overview of the example. The example itself briefly discusseseach module; the pseudocode for that module follows its discussion.
The scenario chosen for this example is a simple terminal session logging utility that uses most of the pseudoterminal capabilities. This example also shows how to use the write-with-echo capability, which provides a significant gain in performance.