Extended OPL Calls

  Extended OPL CallsExtended OPL Calls
  This document is an attempt to list the may possible options available to the OPL-Programmer, which are normally only to be found in the SIBO 'C'-Programming Guides.
  The various tips and tricks are illustrated with examples in Standard OPL.
  There is no particular order to the points listed.  The information can be freely distributed, but is   Psion GmbH.  Thanks to David Wood at Psion PLC for helping me with these routines over the past few months.
  Neither the Author, nor Psion GmbH, takes any responsibility for and damage or loss of data which occurs as result of using information in this document.  The information is subject to change without notice.
                    Comments are shown in italics.
    Use of CALL and OS
  CALLs and OSs take a particular format.  This is briefly explained in the OPL Programming Manual, but is repeated here for reference.
  Using the 'Service' and 'Interrupt' reference in the 'C'-Programming Guide, it is possible to access the specific operating system call desired.
  Format for Call:
     ret%=CALL($xxyy,...,) xx = Service, yy = Interrupt
  Format for OS:
     LOCAL ax%,bx%,cx%,dx%,si%,di% keep these together
     LOCAL flags%
     ax%=$xxyy xx = Service, yy = AL (parameter)
     flags%=OS($zz,addr(ax%)) zz = Interrupt
  If an error occurs when using OS, (flags% AND 1) will be true,
  the error code is given by ax% AND $ff00.
    Starting another process
  Here we start another process, and specify the name of the file to be opened.  The command line starts with 'C' for create, or 'O' for open.  The filename is stored in f$.  The 'Record' at the beginning of the command line in this example is the name of the icon under which the filename will appear in the System Screen.  If this is not a valid (ie. installed) icon, then the filename will appear under RunImg.
  eg. Recorder with a filename
  PROC Record:(f$)
     LOCAL cmdl$(128),helpnm$(128),hpid%,ret%
     cmdl$="CRecord"+chr$(0)+".WVE"+" "+chr$(0)+f$+chr$(0) command line
     call($0688,pid%) ProcResume
  Get Process ID
  Each process has an ID number.  Here we read the ID number for a specific process.
  eg. Get the process ID for the system shell.  pid% is defined globally to hold this variable.
     pid%=call($0188,addr(name$)+1) ProcIdByName
  Getting the Process ID for the current process
  This short piece of code reads the ID for the current process.
     LOCAL pid%
    Setting the priority of the process
  Once we have the ID for a process, we can change the priority of this process.
     LOCAL ax%,bx%,cx%,dx%,si%,di% keep these together
     LOCAL flags% NB: pid% must be defined to be the process id
     flags%=OS($88,addr(ax%)) ProcSetPriority
  Changing the position of a process
  We can also change the position of this process, ie. whether it is in foreground or background.
     For the current process, pid% can be set to zero.
     For foreground, use pos%=0, for background, pos%=100.
    Language Code
  Programs written by Psion are in the majority Multi-lingual.  This means that if they run on an English machine, they run in English, on a German machine they run in German, and so on.
  Each language is assigned a number, which is used to recognise the machine in use.
  This example returns a string containing the number of the code for the resource file.
  PROC Lang$:
     LOCAL ax%,bx%,cx%,dx%,si%,di% Keep these variables together
     LOCAL flags%,a$(2)
     ax%=$1B00 GetLangData
     flags%=OS($008B,ADDR(ax%)) General Services
     IF flags% AND 1
          RETURN("01") an error occured
          IF LEN(a$)<2 :a$="0"+a$ :ENDIF makes two digits for filenames
  The numbers currently in use are:
        1   English
        2   French
        3   German
        4   Spanish
        5   Italian
        6   Swedish
        7   Danish
        8   Norwegen
        9   Finish
        10  American
        11  Swiss French
        12  Swiss German
        13  Portuguese
        14  Turkish
        15  Icelandic
        16  Russian
        17  Hungarian
        18  Dutch
        19  Belgian Flemish
        20  Australian
        21  New Zealand
        22  Austrian
        23  Belgian French
     Simulating a key press
  There is a technique for the Series3a to simulate a keypress in another application.  This method only works for Object-oriented applicaitions.
  The following must be globally defined:
     GLOBAL k%,m% keep these two together
  pid% is the process id, k% is the keycode, m% is the modifier
     CALL($0483,pid%,$31,0,addr(k%)) MessSend
  Capturing a key in background
  We can also when a particular key is pressed, even if the process requiring this key is not current.
     call($c58d,26,$404) wCaptureKey
     This example captures key 26, ie. Ctrl-Z.
  Capturing the off key
  The following parameters will capture the 'OFF' key.
     keya(kstat%,k%(1) queues for a key press
     The value returned in kstat% will NOT be -46 if a key is pressed, and for the off key,
     k%(1) will be $2003.
    Reading an environment variable
  Environment variables are used to store data which is used by all applications.  Space for these variables is limited, and should normally not be used by OPL applications.
  PROC EnvGet:
     LOCAL ax%,bx%,cx%,dx%,si%,di% keep these together
     LOCAL flags%
     LOCAL env$(6),penv%,lenenv%
     LOCAL buff$(255),pbuff%,lenbuff%
     env$="$WP_PW" name of the variable to search for, eg. $WP_PW
     flags%=OS($008b,ADDR(ax%)) GenEnvBufferGet
     IF flags% AND 1
          An error has occured, error code: ax% OR $ff00
          POKEB ADDR(buff$),lenbuff% Insert leading count
  Reading the user details
  User details are stored in environment variable $WS_PW.
  Bytes 4, 8, 12, and 16 contain the length (in bytes) of each of the four lines.
  The first line begins at byte 19.
  Reading the current printer driver
  The printer destination is stored in environment variable P$D. Zero for parallel, one for serial, and two for file.
  The printer driver is stored in environment variable P$M.  This is generally a filename.
  If printing to a file, the name is stored in P$F.
    Asynchronous WVE Playing
  This procedure will play a .WVE file asynchronously.
  PROC Play:(inname$,ticks%,vol%)
     LOCAL name$(128),pstat%
     IOWAITSTAT pstat%
  Cancelling Asynchronous Wave Playing
  PROC Playc:
    Window Server os-calls
  This is a list of the Window Server os-calls, given here for reference purposes only.
  gSetOpenAddress(bx,cx,dx)   $5f8d
  wCancelCaptureKey(bx,cl,ch) $c68d
  wCancelSystemModal(bx) $c88d
  wCaptureKey(bx,cl,ch)  $c58d
  wClientInfo(bx)     $8c8d
  wClientPosition(bx,cx) $998d
  wDisableLeaves(dx)  $0dd6
  wDisablePauseKey()  $ce8d
  wDrawButton(bx,cx,dx)  $608d
  wEnablePauseKey()   $4d8d
  wEndCompute()       $8b8d
  wGetProcessList(si) $d98d
  wSendCommand(bx,cx,dx) $da8d
  wStartCompute()     $8a8d
  wsUpdate(bx)        $528d
  wSystemModal(bx)    $c78d
    Power Status
  This code will return TRUE if the mains is plugged in, and FALSCH if not.
  PROC mainsin%:
     LOCAL esup%(3)
     CALL($118e,addr(esup%(1)))   HwGetSupplyStatus
     RETURN esup%(3)
  Auto-off settings
  This code will return TRUE if the Series3a is NOT going to turn off, eg. because the external power is plugged in.
  PROC notifm%:
     IF (PEEKB(PEEKW($18)+13) and $f <2
          return 0 This is not a 3a
     ELSEIF (CALL($388b) AND $FF) <> 1 GenGetAutoMains
          return 0
  Now we can read the number of seconds to go before the Auto-off turns the machine off.  Note: In OPL  this will only work if we first call GenMarkNotActive and wEndCompute.
     ie.  CALL($138b) GenMarkNotActive
          CALL($8b8d) wEndCompute
  PROC timelft%:
     LOCAL gooff%,timeoff% keep these together
     IF notifm%:
          RETURN -1
     CALL($078b,0,4,0,$40a,ADDR(gooff%)) GenGetOsData
     IF gooff%<0
          RETURN -1
    Asynchronous reading
  It is possible to read keys and timers asynchronously.  Firstly, a key:
  PROC quekey:
  kstat% and k%(2) are globally defined.  Kstat% will by -46 if no key was pressed.  If the off key is pressed (and has been captured), then k%(1) will be $2003, otherwise the values in k%() will be similar to those returned using GETEVENT.
  Now the timer:
  Firstly we need to globally define timh%, timstat%, and then we open the channel to the timer function.
  Now we can queue the timer.
  PROC quetim:
     LOCAL t&,t%
     t%=2 This is the number of seconds before the timer should expire
    Further Reading
  This has been only a brief guide to some of the functions available on the Series3a.  For more information, see the SIBO 'C'-Programming Guide.
No Comments
Back to top