The mx routines are classified into four groups:
The first group includes routines that open and close table files. A table must be opened before it can be read or updated. Opening a table will also lock it at the level indicated in the data dictionary for the database. Any tables which have been opened should be closed before the program ends, and this will also unlock them.
The second group of routines is used to specify a subset of a table. This is analogous to the construction of a WHERE clause in the Query Language.
The third group of routines is used to retrieve records from one or more tables. Any qualifications which may have been constructed are used by these routines. The retrieval routines will automatically use any appropriate indices present to improve retrieval speed.
The fourth group is used to add to, delete from, and update tables.
The header file mscc.h must be included at the beginning of each program using the mx routines. It has the correct definitions for all the mx (and mr) routines and a number of useful definitions of data types and constants. mscc.h also includes the Standard I/O Library, so there is no need to include this in your programs. The file is shown below.
Please note that the mscc.h file differs from version to version. For an exact reference, the file is located in the following $EMPRESSPATH/include directory.
/*********************************************************************** * (c) Copyright Empress Software Inc. 1983, 2003 ***********************************************************************/ #ifndef __MSCC_H #define __MSCC_H #ifdef __cplusplus extern "C" { #endif #include <stdio.h> #include "c.h" #include <misc/public.h> #include <usrmp/msdsqldt.h> #include <usrfns/dtparvar.hx> #include <api/mrapi.hx> #include <api/msapi.hx> #include <api/mxapi.hx> #include <misc/init.hx> #include <misc/mscc_com.h> #include <misc/mxe.h> #include <misc/mxtrig.h> #ifdef __cplusplus } #endif #endif /* __MSCC_H */
There are two ways to organize a program using the Empress Host Language Interface routines:
You can name the regular main procedure msmain. This will include a default main procedure which does the necessary initializations. msmain may have the parameters argc and argv as usual. Similarly, name the regular C exit procedure msexit.
You can write your own main procedure, calling msinit to begin and msexit to end. These two procedures carry out various initialization and cleanup tasks.
The msinit can be called several times during the execution of a program. Subsequent msinit calls should be preceeded, in general case, by a cleanup task (i.e. msend). However, this is not neccessary, and spawned processes can invoke msinit although msend hasn't been called previously.
The routine msclean allows you to put a procedure on the stack to be called immediately before the program exits.
Programs using the mx routines must be compiled using empcc or empc++. The effect of this is to ensure that Empress library routines are included in the compilation; in all other respects, for example, empcc behaves similarly to cc.
Note that mscall statements may also be used in programs compiled by empcc or empc++. For example, the simplest way to create a new table in an application program is to use an mscall statement.
Throughout this manual, references will be made to internal, external, and file formats of attribute values.
The external format is what is printed by a Query Language SELECT command. External format values are character strings.
The internal format is the format in which the value is manipulated by Empress, and does not necessarily correspond to the external format. For example, date (1) values are not manipulated as (for instance) "1 May 1981'', nor are dollar values manipulated complete with dollar signs, asterisks, and commas.
Finally, there is the file format, which is the way the value is actually stored by Empress. Application programs do not retrieve file format values.
Attribute values are normally retrieved in external format, though it is possible to retrieve integers and internal format values directly. Values are normally assigned to attributes in external format or as integers, though it is possible to assign an internal format value directly to an attribute.
To do arithmetic on external format (character string) retrieved values, they must first be converted to numeric values using sscanf, atof, etc. Note that dollar signs, asterisks, and so on must be removed from dollar values before converting them. A simple procedure for doing this, dollcvt, is presented in the example program interest.c in the next chapter.
Bulk data values are handled differently from other types. To output bulk values in external format (as in a Query Language SELECT) would require converting the bytes to hexadecimal characters, which is not what is generally needed. Instead, bulk data is normally output in internal format using the routine mxcopyi. This routine is intended specifically for retrieving bulk data, though it can be used to get the internal format of any other type of data as well. The routine mxputi is intended for placing bulk data into a record.
There are situations in which the routines are properly unable to execute successfully, such as when record level locking is in force for a table and a record which the routine needs to access is locked, or when the data which a routine attempts to insert into a table is disallowed because of range checking or a limited duplicate index, or when inappropriate data is passed to one of the qualification routines.
Most of the mx routines return a value of true or false, depending on whether they succeed. What happens on failure depends on the value of the variable mxerret. If this is 0, the calling program is terminated with a suitable error message; if it is 1, the calling program is not terminated, the variable mxoperr is set to one of the MRER... codes listed in the header file mrerrno.h, and control is returned to the program. mxoperr may be examined by the application program to determine why the routine fails, and something sensible done about it.
The default value for mxerret is 0, so application programs which do something about failures must set it to 1.
The mx routines that do not return true or false print an error message and terminate the calling program when they fail.
The mrerrno.h is listed below. Please note that this file differs from version to version. For an exact reference, the file is located in the $EMPRESSPATH/include/usrmr directory.
/*********************************************************************** * (c) Copyright Empress Software Inc. 1983, 2003 ***********************************************************************/ #define MRNOERR 0 /* no error indication*/ #define MRERNAME 1 /* invalid table name */ #define MREREXIST 2 /* table already exists */ #define MRERNEXIST 3 /* table does not exist */ #define MRERRPERM 4 /* no read permission on table */ #define MRERWPERM 5 /* no read/write permission on tab */ #define MRERBUSY 6 /* database/table lock busy */ #define MRERVWOP 7 /* operation invalid on view */ #define MRERVWDEF 8 /* invalid definition for view */ #define MRERVWRECUR 9 /* recursive definition for view */ #define MRERDBEXIST 10 /* not a database */ #define MRERANAME 11 /* invalid attribute name */ #define MRERAEXIST 12 /* attribute already exists */ #define MRERANEXIST 13 /* attribute does not exist */ #define MRERDTPAR 14 /* invalid data type/parameters */ #define MRERINAME 15 /* invalid index name */ #define MRERNULL 16 /* null value for attribute */ #define MRERUDUP 17 /* duplicate value for attribute */ #define MRERDUP 18 /* too many duplicate values */ #define MRERCVT 19 /* bad value conversion for attr */ #define MRERPRIV 20 /* no privilege for table */ #define MRERAPRIV 21 /* no update privilege on attribute */ #define MRERCRPRIV 22 /* no create table privilege */ #define MRERQOP 23 /* invalid operator/function */ #define MRERQATTR 24 /* incompatible attributes */ #define MRERQPAT 25 /* invalid pattern */ #define MRERQRNG 26 /* invalid range */ #define MRERXCONS 27 /* invalid constant */ #define MRERXCVT 28 /* invalid conversion */ #define MRERXFUNC 29 /* invalid operator/function */ #define MRERIEXIST 30 /* index already exists */ #define MRERINEXIST 31 /* index does not exist */ #define MRERICOMP 32 /* index name/definition mismatch */ #define MRERCREATOR 33 /* invalid creator name */ #define MRERCRPERMS 34 /* invalid create permissions */ #define MRERDDEXIST 35 /* data dictionary already exists */ #define MRERDBA 36 /* not the dba */ #define MRERINPRIV 37 /* insufficient privileges */ #define MRERDBACR 38 /* not the dba or creator */ #define MRERRCNAME 39 /* invalid constraint name */ #define MRERRCEXIST 40 /* constraint already exists */ #define MRERRCNEXIST 41 /* constraint does not exist */ #define MRERRCINSERT 42 /* insert constraint violation */ #define MRERRCDELETE 43 /* delete constraint violation */ #define MRERRCLKBUSY 44 /* cannot check constraint */ #define MRERRCINCDTS 45 /* incompatible types in constraint */ #define MRERRCOTABLE 46 /* cannot open table for constraint */ #define MRERRCNATTR 47 /* invalid attribute in constraint */ #define MRERNLOC 48 /* invalid location */ #define MRERVWTWICE 49 /* table selected twice in view */ #define MRERVWNONEQ 50 /* join condition without '=' */ #define MRERVWOROP 51 /* OR conjunction in view */ #define MRERVWNOTOP 52 /* NOT operator in view */ #define MRERVWJOIN 53 /* insufficient join conditions */ #define MRERVWKEY 54 /* insufficient keys in view */ #define MRERVWREATTR 55 /* view attributes repeated/equated */ #define MRERVWNONULL 56 /* cannot obtain 'not null' value */ #define MRERVWUNIND 57 /* unique value unobtainable in view */ #define MREREXPR 58 /* expression failure */ #define MRERTRPROGRESS 59 /* transaction already in progress */ #define MRERTRSTARTNOT 60 /* unable to start transaction */ #define MRERTROFF 61 /* no transaction in progress */ #define MRERTRINVSVP 62 /* invalid save point name */ #define MRERTRSVPEXISTS 63 /* save point name already exists */ #define MRERTRNOSVP 64 /* save point does not exist */ #define MRERNUPDMODE 65 /* table opened in non-updatable mode */ #define MRERSERVER 66 /* server not available */ #define MRERANOTDTF 67 /* not a variable length attribute */ #define MRERINOVRFL 68 /* index has no overflow file */ #define MRERCRFL 69 /* cannot create data files */ #define MRERCRFLACCESS 70 /* no access to create data files */ #define MRERVWATTREX 71 /* too many attribute names specified */ #define MRERGENDT 72 /* unknown generic type */ #define MRERRECNO 73 /* invalid record number */ #define MRERDT 74 /* invalid data type */ #define MREREXDTPAR 75 /* too many parameters for the datatype */ #define MRERBADDTPAR 76 /* bad datatype parameter */ #define MRERATTRSPEC 77 /* attribute specification not allowed */ #define MRERHAVBADA 78 /* disallowed attribute in having clause */ #define MRERVWGRPGRP 79 /* no group by allowed on grouped view */ #define MRERVWGRPHAV 80 /* no having allowed on grouped view */ #define MRERVWGRPJOIN 81 /* no join allowed on a grouped view */ #define MRERVWGRPQUAL 82 /* no qual allowed on grouped view */ #define MRERVWORDERINT 83 /* invalid order by specification (integer) */ #define MRERVWINSELHAV 84 /* invalid select list with having clause */ #define MRERQNESTATTR 85 /* too many attributes in nested select */ #define MRERVWAGGATTRX 86 /* no derived aggr attribute in expression */ #define MRERVWAGGRHAV 87 /* no having allowed on aggregate view */ #define MRERVWAGGRJOIN 88 /* no join allowed on a aggregate view */ #define MRERVWAGGRQUAL 89 /* no qual allowed on aggregate view */ #define MRERVWGRPAGGR 90 /* no aggr function allowed on grouped view */ #define MRERVWINSELDIST 91 /* invalid select list with select distinct */ #define MRERVWAGGATTRG 92 /* no derived aggr attribute in group by */ #define MRERNOHASHREC 93 /* record not found in table */ #define MREROPMODE 94 /* invalid table open mode */ #define MRERVWORDERSEL 95 /* invalid order by specification (select) */ #define MRERINVTXT 96 /* invalid char in the value for attribute */ #define MRERDRIOATTR 97 /* direct I/O not supported for attribute */ #define MRERDRIOSET 98 /* file redirection error */ #define MRERCRSIZE 99 /* record size too large */ #define MRERNLCOMP 100 /* null compile entry for the table */ #define MRERNOCOMP 101 /* no compile entry for the table */ #define MRERNVEW 102 /* expecting a view, got a table */ #define MR_DB_ALREADY_LOCKED 103 /* the database has been already locked */ #define MR_IN_ADM 104 /* the dba put the database down*/ #define MRERDTER 105 /* Refer to error at DT level */ #define MRERINVUREC 106 /* invalid record for update */ #define MRERURECDEL 107 /* record for update has been deleted */ #define MRERAGGRNORECS 108 /* no records to do aggregate */ #define MRERNOCRSHM 109 /* cannot create shared memory */ #define MRERNORMSHM 110 /* cannot remove shared memory */ #define MREROUTERILL 111 /* Illegal outer join */ #define MRERNOSHM 112 /* No shared memory available */ #define MRERDRECDEL 113 /* deleted record for delete */ #define MRERNONEXIST 114 /* no data file for '%s' */ #define MRERNET 115 /* network error in file operation */ #define MREROPEN 116 /* bad data file for '%s' */ #define MRERFLOPMODE 117 /* invalid open mode '%c' */ #define MRERSVPORT_BADID 118 /* invalid server port id */ #define MRERSVPORT_INCOMPAT 119 /* incompatible server port id */ #define MRERSVPORT_UNDEF 120 /* server port not defined */ #define MREROPDICTMODE 121 /* invalid data dictionary open mode */ #define MRERFULLBUSY 122 /* lock resource is full */ #define MRERNOLKRESOURCE 123 /* cannot access lock resource */ #define MRERBADRECORD 124 /* bad record retreived */ #define MRERBADDATARETR 125 /* from DT: bad var.len.val.retreived*/ #define MRERSORTLOCK 126 /* sort by error(locked) **WARNING */ #define MRERMNAME 127 /* invalid module name */ #define MRERMEXIST 128 /* module already exist */ #define MRERMNEXIST 129 /* module does not exist */ #define MRERLANGOPTION 130 /* invalid language option */ #define MRERFNAME 131 /* invalid SQL-invoked routine name */ #define MRERFMCLASS 132 /* there are multiple function classes for same routine */ #define MRERFEXIST 133 /* SQL-invoked routine already exist */ #define MRERFNEXIST 134 /* SQL-invoked routine not exist */ #define MRERPARAMNAME 135 /* invalid parameter name */ #define MRERPARAMEXIST 136 /* parameter already exist */ #define MRERPARAMNEXIST 137 /* parameter does not exist */ #define MRERPARAMNO 138 /* invalid parameter number */ #define MRERQPARAM 139 /* incompatible parameter */ #define MRERPARAMNUM 140 /* too many parameters */ #define MRERPARAMNUM1 141 /* expecting one parameter */ #define MRERPARAMNUM2 142 /* expecting two parameters */ #define MRERPARAMB 143 /* parameter data type should be generic boolean */ #define MRERPARAMNB 144 /* parameter data type should be parameteric data type */ #define MRERRETB 145 /* return data type should be generic boolean */ #define MRERRETNB 146 /* return data type should be parametric data type */ #define MRERARETNB 147 /* aggregate function's return data type should be parametric data type */ #define MREREXTNAME 148 /* invalid external function name */ #define MREREXTEXIST 149 /* external name already exist */ #define MRERFILENEXIST 150 /* file doesn't exist */ #define MRERTRNAME 151 /* invalid trigger name */ #define MRERTREXIST 152 /* trigger already exist */ #define MRERTRNEXIST 153 /* trigger does not exist */ #define MRERMODWRAPCOMPFAIL 154 /* failed to compile wrapper for module */ #define MRERDDCACHEFAIL 155 /* failed to modify DD_CACHE for module */ #define MRERXSVFUNC 156 /* SQL-invoked routine '%s' is not available on the client */ #define MRERTRIGABORT 157 /* operation aborted by trigger */ #define MRERXTRIGPROC 158 /* trigger procedure should have no parameter */ #define MRERNOTRIGPROC 159 /* trigger procedure '%s' unavailable */ #define MRERTRIG_OLD_NEW 160 /* attribute '%s' must be qualified by OLD or NEW */ #define MRERTRIGNESTED 161 /* nested queries are not allowed for triggers */ #define MRERPARAMSTYLE 162 /* parameter style of SQL-invoked routine '%s' inconsistent */ #define MRERMODWRAPGENFAIL 163 /* failed to generate wrapper for system */ #define MRERNOATTR 164 /* no attributes left in table '%s */ #define MRERRNGCONVT 165 /* cannot convert range check for attribute '%s' */ #define MRERCANTADDATTR 166 /* cannot add non-null attribute to non-empty table */ #define MRERDUPATTR 167 /* duplicate attribute '%s' */ #define MRERFEWCONS 168 /* too few constants */ #define MRERMANYCONS 169 /* too many constants */ #define MRERWRONGCONS 170 /* wrong number of constants */ #define MRERAINVEXPR 171 /* invalid expression for attribute '%s' */ #define MRERSAMETAB 172 /* same table involved in select and insert */ #define MRERFEWSETATTR 173 /* too few SET attributes */ #define MRERMANYSETATTR 174 /* too many SET attributes */ #define MRERFEWATTRS 175 /* too few attributes in '%s' */ #define MRERNOTNULLATTR 176 /* not-null attribute '%s' is not included in attribute list */ #define MRERINVATTR 177 /* invalid attribute '%s' for table '%s' */ #define MRERINVIDXPARM 178 /* invalid index parameters (%s, %s)" */ #define MRERNOPRIVCRVW 179 /* insufficient privileges to create view */ #define MRERUNEQCONSTR 180 /* unequal number of attributes in the constraint */ #define MRERNOCONSTR 181 /* constraint '%s' does not exist */ #define MRERNOIDX 182 /* index '%s' does not exist */ #define MRERSYSTABLE 183 /* cannot perform %s operation on %s */ #define MRERINVOPR 184 /* invalid operator '%s' */ #define MREREXPRPROB 185 /* expression cannot be evaluated or out of range */ #define MRERUPDMOD 186 /* failed to update module '%s' (unable to resolve symbol) */ #define MRERCOPYRL 187 /* failed to copy file from server to client */ #define MRERCOPYLR 188 /* failed to copy file from client to server */ #define MRERINVCALLEXPR 189 /* invalid call expression, output is not bound with a variable */ /* next number is 190 */
If you need to open your own files in your program and cannot do so because Empress has used all available file descriptors, you will have to limit the number of files Empress will attempt to open. This can be done by setting the Empress system variable MSFILESOPEN in the operating system. For example, to limit the number of files Empress will attempt to open to 10:
setenv MSFILESOPEN 10
MSFILESOPEN = 10 export MSFILESOPEN
set MSFILESOPEN = 10
Not that, this will not prevent Empress from accessing more than 10 files. Empress will close the least-recently used file before opening another file, if the quota is reached. If the file that was closed by Empress is accessed again, Empress will automatically re-open it. Empress keeps track of these files internally and the closing and re-opening of files are transparent to the user.
For details on setting Empress system variables, refer to the Setting Variables in the Empress SQL: User's Guide.
The Empress mx routines have been named in a logical manner to indicate their functions. Since external names in C are limited, the mx routine names are by necessity somewhat abbreviated. Each routine name begins with "mx''. Most then have a third generic letter, such as "q'' for the routines dealing with qualifications, or a full word, such as "get'' for the routines which find attribute values. A classification and list of all the mx routines is shown on the next page. Manual pages with detailed specifications for each of these routines may be found at the end of the manual, and should be referred to when writing programs.
The following table lists the routines used to open and close tables:
Table 2-1: Open and Close Routines
Function | Purpose |
mxopen | Open a table. |
mxclose | Close a table. |
mxaclose | Close all tables and de-allocates storage. |
The following table lists the routines used to check attributes:
Table 2-2: Check Attributes Routines
Function | Purpose |
mxchkattr | Check whether an attribute exists. |
mxigeta | Find an attribute name given its number. |
The following table lists the routines used to retrieve attribute values:
Table 2-3: Retrieval Routines
Function | Purpose |
mxgetbegin | Initialize retrieval routines |
mxsrtbegin | Initialize retrieval routines for sorted output. |
mxget | Retrieve the next record. |
mxreget | Try again to retrieve the next record. |
mxprev | Retrieve the previous record. |
mxreprev | Try again to retrieve the previous record. |
mxgetend | Terminate retrieval routines |
mxgetvs | Get an external format attribute value. |
mxgetvi | Get an integer attribute value. |
mxcopyi | Get an internal format attribute value. |
The following table lists the routines used to modify attribute and record values:
Table 2-4: Update Routines
Function | Purpose |
mxadd | Insert a record into a table. |
mxdel | Delete a record from a table. |
mxput | Update a record in a table. |
mxputvs | Put an external format value into an attribute. |
mxputvi | Put an integer value into an attribute. |
mxputi | Put an internal format value into an attribute. |
mxsetnull | Set all attributes in a record to NULL. |
The following table lists the routines used to build qualifications for data retrieval:
Table 2-5: Qualification Routines
Function | Purpose |
mxqcon | Compare an attribute with an external format constant. |
mxqconi | Compare an attribute with an internal format constant. |
mxqrng | Compare an attribute with a range (external format). |
mxqrngi | Compare an attribute with a range (internal format). |
mxqatr | Compare two attribute values. |
mxqmch | Match an attribute value with a pattern. |
mxqnul | Compare an attribute value with NULL. |
mxqand | Perform an AND on two qualifications. |
mxqor | Perform an OR on two qualifications. |
mxqnot | Negate a qualification. |
mxqieq | Qualify attributes equal to an integer value. |
mxqseq | Qualify attributes equal to a string value. |
mxcompare | Test if a constant is "=", ">", or "<" an attribute value. |
The following table lists the routines used to control transactions:
Table 2-6: Transaction Routines
Function | Purpose |
mxtrstart | Start a transaction. |
mxtrcancel | Cancel a transaction. |
mxtrcommit | Commit a transaction. |
mxtrsave | Set a save point. |
mxtrrollback | Roll back to save point. |
When writing applications programs, if one of your own procedures has the same name as one of the Empress internal procedures, problems occur (e.g., complaints from the loader about multiply-defined routines). To help spot this problem, the prefixes used in naming the Empress internal procedures are listed below. (This is not a complete list and it may change from version to version of Empress.) If you are having problems with one of your routines and its name begins with one of these prefixes, try changing its name.
addr | ap* | ||||||||
boolean | byte | ||||||||
cpbytes* | cv* | ||||||||
date* | day* | dt* | |||||||
er* | ewrite* | exit* | extsrt* | ||||||
fl* | fm* | fork* | |||||||
gdat* | gdec* | gflt* | gint* | ||||||
ll* | |||||||||
main | match* | mem* | month* | mp* | mr* | ms* | mw* | mx* | |
of* | |||||||||
pos | prog* | ||||||||
ql* | |||||||||
ran* | |||||||||
sc* | sep* | sfmt | sig* | sm* | sp* | str* | |||
taddr | time* | tty* | |||||||
ufx* | |||||||||
varg* | |||||||||
wkday* | word | ww* | |||||||
xf* | |||||||||
year* |
Given a database directory, a table within it, and a code for the required open mode, mxopen opens the table and locks it at the level indicated in the data dictionary for the database.
The presence or absence of a particular attribute within a table may be checked by the routine mxchkattr.
Record retrievals must be initialized by calling mxgetbegin, which associates any qualification (an internal representation of a WHERE clause) with the record. (mxgetbegin must be called even if there is no qualification.) If sorting is desired, mxsrtbegin should be used instead of mxgetbegin.
Retrievals are accomplished by repeated calls to mxget and mxprev. Indices will automatically be used to improve retrieval speed if they are present. mxgetvs is used to transfer an attribute value to user working storage. This produces a character string representation of the data, which may be examined and manipulated by the user's program. mxgetend must be called to clean up after retrievals.
To create a new record, attribute values are added to a record using mxputvs, and an entire record may be set to NULL using mxsetnull. The record is then added to the table by a call to mxadd.
Records are deleted using mxdel.
Records are updated using mxputv to change attribute values in a previously-retrieved record, and mxput to replace the old record by the new one.
Qualifications are created by calls to mxqcon, mxqconi, mxqrng, mxqrngi, mxqatr, mxqmch, mxqnul, mxqieq, or mxqseq, which compare an attribute value with a constant, a range of values, or another attribute value, perform a match with a constant, a comparison with NULL, an "equals'' comparison with an integer, or an "equals'' comparison with a string, respectively. Complex qualifications are produced using mxqand, mxqor, or mxqnot, which perform AND and OR functions and negate another qualification, respectively.
If you simply wish to see whether a value in external format is equal to, less than, or greater than an attribute value, the routine mxcompare may be used.
When all operations on a table have been finished, it should be closed using mxclose. Note that it is good practice to close tables once they are no longer needed; otherwise, the application may run out of space.
If an application program terminates abnormally due to some error condition, such as receiving an interrupt, the routine mxaclose may be called to close all open tables and de-allocate all mx-used storage. This routine should not be used indiscriminately in place of mxclose and mxgetend, however, as using it defeats all the normal error-checking done by the mx routines. (If an application program has failed to close tables or de-allocate storage, an error message warning that space is still allocated is normally printed when the program is run.)
A transaction may be started by mxtrstart, committed by mxtrcommit, or cancelled by mxtrcancel. (Note that a transaction may not be started, committed, or cancelled in the middle of a retrieval loop.) A save point in a transaction is set with mxtrsave and a transaction is rolled back to a save point with mxtrrollback.
Most arguments to the mx routines are used to specify database, table, or attribute names, or data values. Generally, these arguments are character strings.
When two or more opened databases have tables with the same name, the database name and a colon are prefixed to the table name, thus:
database:table
Attribute names can be similarly qualified by prefixing a table name and a period to the attribute name. This may be further qualified by a database name as before. Hence, acceptable forms of attribute names are:
attribute table.attribute database:table.attribute
It is good practice to qualify attribute names with their table names as a matter of course, since errors resulting from dealing with an attribute in the wrong table may be quite difficult to spot when debugging code. This also guards against confusion resulting from later attribute name changes in the tables used by the application.
Several Empress-related errors are liable to occur when using the mx routines. There are five major types of errors:
All have the following format:
*** Error Type *** explanation_of_error
Program Bug
A program bug error message indicates that there is a programming mistake in the applications program, and usually arises from passing a routine the wrong parameters. As another example, attempting to insert data into a table which was opened in read-only mode will also cause program bug errors.
Space Problem
This kind of error simply indicates that the program has run out of space. If you get this error, check to see that any open tables are closed once you no longer need them.
User Error
This kind of error is not strictly related to the use of the mx routines themselves, but is more likely to result from data input by the users who are running the applications. It is generally caused by passing incorrect table or attribute names to a routine. Incorrect names which are written into an applications program are usually easy to find and fix. The routine mxchkattr can be used to check the validity of attribute names.
Database Problem
This error is very serious, and occurs when the database is found to be corrupted.
File Problem
This error occurs when the mx routines cannot access a file which they should be able to access. It is generally a serious problem, and may indicate a corrupted database.
If locking is specified for a table in the data dictionary of the database, the table or its records can be locked in read or update mode. If no lock level is specified in the data dictionary, no locks can be placed on the table/records.
Locking a table or record in read mode means that others may access it, provided they also access it in read mode. Thus, anyone who wishes to update this table or record will be prevented from accessing it until everyone is finished with the table.
Locking a table or record in update mode means no one else may access it for reading or updating. This can cause a bottleneck if one person keeps the table or record locked for a long period of time, or if a table or record is locked and never unlocked.
A table can be opened in read, update or deferred mode. Opening a table in read mode places read locks on the table. Opening a table in update mode places update locks on the table. Opening a table in deferred mode places read locks on the table initially, but these locks are exchanged for update locks when the table is written to.
A table level lock takes effect when a table is opened. A group level lock takes effect when an mxgetbegin is executed. A record level lock takes effect when an mxget is executed.
Some applications, especially those involving a read-only database or a "single-user'' database, do not require locks. These applications are better implemented without using locks, since employing the locking mechanism introduces complexities and is hence more error-prone. The Database Administrator should set guidelines for the use or non-use of locks, and set the default locking on tables accordingly. For a discussion on locking, see the Empress SQL: Reference manual under the LOCK LEVEL command.
The utility program empclrlk and empadm may be used to clean up any hanging locks on a database. This is particularly useful when debugging programs with MSDEBUG set to abort, because if a program core dumps you will almost certainly have left-over locks which should be removed.
Executable programs containing calls to mx routines can be moved across machines. However, execution errors will result if the installed Empress path is different across machines. In this case, programs should be re-compiled on the target machine.