tmi2_fluffos_v2/
tmi2_fluffos_v2/bin/
tmi2_fluffos_v2/etc/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/ChangeLog.old/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/Win32/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/compat/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/compat/simuls/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/include/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/clone/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/command/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/data/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/etc/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/include/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/inherit/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/inherit/master/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/log/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/single/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/single/tests/compiler/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/single/tests/efuns/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/single/tests/operators/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/testsuite/u/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/tmp/
tmi2_fluffos_v2/fluffos-2.7-ds2.018/windows/
tmi2_fluffos_v2/lib/
tmi2_fluffos_v2/lib/adm/
tmi2_fluffos_v2/lib/adm/daemons/languages/
tmi2_fluffos_v2/lib/adm/daemons/network/I3/
tmi2_fluffos_v2/lib/adm/daemons/virtual/
tmi2_fluffos_v2/lib/adm/daemons/virtual/template/
tmi2_fluffos_v2/lib/adm/news/
tmi2_fluffos_v2/lib/adm/obj/
tmi2_fluffos_v2/lib/adm/obj/master/
tmi2_fluffos_v2/lib/adm/priv/
tmi2_fluffos_v2/lib/adm/shell/
tmi2_fluffos_v2/lib/adm/tmp/
tmi2_fluffos_v2/lib/cmds/
tmi2_fluffos_v2/lib/d/
tmi2_fluffos_v2/lib/d/Conf/
tmi2_fluffos_v2/lib/d/Conf/adm/
tmi2_fluffos_v2/lib/d/Conf/boards/
tmi2_fluffos_v2/lib/d/Conf/cmds/
tmi2_fluffos_v2/lib/d/Conf/data/
tmi2_fluffos_v2/lib/d/Conf/logs/
tmi2_fluffos_v2/lib/d/Conf/obj/
tmi2_fluffos_v2/lib/d/Conf/text/help/
tmi2_fluffos_v2/lib/d/Fooland/adm/
tmi2_fluffos_v2/lib/d/Fooland/data/
tmi2_fluffos_v2/lib/d/Fooland/data/attic/
tmi2_fluffos_v2/lib/d/Fooland/items/
tmi2_fluffos_v2/lib/d/TMI/
tmi2_fluffos_v2/lib/d/TMI/adm/
tmi2_fluffos_v2/lib/d/TMI/boards/
tmi2_fluffos_v2/lib/d/TMI/data/
tmi2_fluffos_v2/lib/d/TMI/rooms/
tmi2_fluffos_v2/lib/d/grid/
tmi2_fluffos_v2/lib/d/grid/adm/
tmi2_fluffos_v2/lib/d/grid/data/
tmi2_fluffos_v2/lib/d/std/
tmi2_fluffos_v2/lib/d/std/adm/
tmi2_fluffos_v2/lib/data/adm/
tmi2_fluffos_v2/lib/data/adm/daemons/
tmi2_fluffos_v2/lib/data/adm/daemons/doc_d/
tmi2_fluffos_v2/lib/data/adm/daemons/emoted/
tmi2_fluffos_v2/lib/data/adm/daemons/network/http/
tmi2_fluffos_v2/lib/data/adm/daemons/network/services/mail_q/
tmi2_fluffos_v2/lib/data/adm/daemons/network/smtp/
tmi2_fluffos_v2/lib/data/adm/daemons/news/archives/
tmi2_fluffos_v2/lib/data/attic/connection/
tmi2_fluffos_v2/lib/data/attic/user/
tmi2_fluffos_v2/lib/data/std/connection/b/
tmi2_fluffos_v2/lib/data/std/connection/l/
tmi2_fluffos_v2/lib/data/std/user/a/
tmi2_fluffos_v2/lib/data/std/user/b/
tmi2_fluffos_v2/lib/data/std/user/d/
tmi2_fluffos_v2/lib/data/std/user/f/
tmi2_fluffos_v2/lib/data/std/user/l/
tmi2_fluffos_v2/lib/data/std/user/x/
tmi2_fluffos_v2/lib/data/u/d/dm/working/doc_d/
tmi2_fluffos_v2/lib/data/u/l/leto/doc_d/
tmi2_fluffos_v2/lib/data/u/l/leto/smtp/
tmi2_fluffos_v2/lib/doc/
tmi2_fluffos_v2/lib/doc/driverdoc/applies/
tmi2_fluffos_v2/lib/doc/driverdoc/applies/interactive/
tmi2_fluffos_v2/lib/doc/driverdoc/concepts/
tmi2_fluffos_v2/lib/doc/driverdoc/driver/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/arrays/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/buffers/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/compile/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/ed/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/filesystem/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/floats/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/functions/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/general/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/mappings/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/numbers/
tmi2_fluffos_v2/lib/doc/driverdoc/efuns/parsing/
tmi2_fluffos_v2/lib/doc/driverdoc/lpc/constructs/
tmi2_fluffos_v2/lib/doc/driverdoc/lpc/preprocessor/
tmi2_fluffos_v2/lib/doc/driverdoc/lpc/types/
tmi2_fluffos_v2/lib/doc/driverdoc/platforms/
tmi2_fluffos_v2/lib/doc/mudlib/
tmi2_fluffos_v2/lib/ftp/
tmi2_fluffos_v2/lib/include/driver/
tmi2_fluffos_v2/lib/log/
tmi2_fluffos_v2/lib/log/driver/
tmi2_fluffos_v2/lib/obj/net/
tmi2_fluffos_v2/lib/obj/shells/
tmi2_fluffos_v2/lib/obj/tools/
tmi2_fluffos_v2/lib/std/adt/
tmi2_fluffos_v2/lib/std/board/
tmi2_fluffos_v2/lib/std/body/
tmi2_fluffos_v2/lib/std/fun/
tmi2_fluffos_v2/lib/std/living/
tmi2_fluffos_v2/lib/std/object/
tmi2_fluffos_v2/lib/std/shop/
tmi2_fluffos_v2/lib/std/socket/
tmi2_fluffos_v2/lib/std/user/
tmi2_fluffos_v2/lib/std/virtual/
tmi2_fluffos_v2/lib/student/
tmi2_fluffos_v2/lib/student/kalypso/
tmi2_fluffos_v2/lib/student/kalypso/armor/
tmi2_fluffos_v2/lib/student/kalypso/rooms/
tmi2_fluffos_v2/lib/student/kalypso/weapons/
tmi2_fluffos_v2/lib/u/l/leto/
tmi2_fluffos_v2/lib/u/l/leto/cmds/
tmi2_fluffos_v2/lib/www/errors/
tmi2_fluffos_v2/lib/www/gateways/
tmi2_fluffos_v2/lib/www/images/
tmi2_fluffos_v2/old/
tmi2_fluffos_v2/win32/
0.9.18 to 0.9.19
 
. COMPAT BUSTER: array = array + scalar is no longer allowed
  (array += scalar was already not allowed)
 
. COMPAT BUSTER: socket_accept() no longer assumes the newly created socket
  is immediately writable; this may cause slightly different behavior (such
  as the return of EECALLBACK on the first socket_write() whereas it never
  happened before, or write_callbacks being called that never got called
  before)
 
. COMPAT BUSTER: socket callbacks (including the resolve() callback) can
  now be static and will still be callable by the driver.  this can cause
  different previous_object() behavior, causing some old socket callback
  security checks to behave improperly
 
. COMPAT BUSTER: fixed explode() so that explode("", "\n") does not return
  ({ 0 })...and removed support for a string as first argument of
  move_object().
 
. COMPAT BUSTER: removed the following efuns: cat(), log_file(), extract(),
  shadowp(), next_living(), add_worth(), add_verb(), and add_xverb()
 
. COMPAT BUSTER: removed the 'cost' and 'worth' fields from 
  domain/author_stats
 
. functionp() now returns 2 instead of 1 in the case where 
  f = (: obj, ({ str, arg0, arg1, arg2, ... }) :).  (ie, the function portion 
  is an array)
 
. geteuid() may now be used on type 'function'.  geteuid(f) where f is of
  type function will return the euid of the constructor of 'f'.  if the
  constructor didn't have an euid at the time of f's creation, its uid
  is stored instead
 
. ed's indentor ('I' command) fixed
 
. fixed the usage of '!' in get_char() mode
 
. fixed several Linux crash/hang bugs
 
. mud_status() improved, total now matches up with the sum of the fields
 
. revised parse_command() and process_string() efuns, added process_value()
  and break_string() efuns (see docs for information)
 
. added a new LPC type called 'buffer' that is a cross between the
  LPC array type and the LPC string type.  'buffer' is intended as a way to 
  conveniently manipulate binary data.  'buffer' is not zero-terminated 
  (that is, it has an associated length).  A 'buffer' is an array of bytes 
  that is implemented using one byte per element.  buf[i] = x and x = buf[i] 
  are allowed and do work.  sizeof(buf) works.  bufferp(buf) is available.  
  buf[i..j] should work as well.  buff = read_buffer(file_name, ...) 
  (same args as read_bytes).  also 'int write_buffer(string file, int start, 
  mixed source)', buf = buf1 + buf2; buf += buf1, buf = allocate_buffer(size).
  The socket efuns have been modified to accept and return the 'buffer' type.  
  Two new socket modes are available:  STREAM_BINARY (3), and 
  DATAGRAM_BINARY (4).  the runtime config file now needs a 'maximum buffer
  size' parameter in it which sets the largest 'buffer' that may be created.
 
. now possible to use write_bytes() to write past the end of the existing file
 
. read_bytes(filename) will now read the entire file (instead of requiring 
  read_bytes(filename, 0, length_of_file)).
 
. added an 'int crc32(buffer b|string s) efun which computes a 32-bit
  cyclic redundancy code for a given buffer or string.  This is the same 
  algorithm (and code) that Zmodem uses for doing its CRC calculations.
 
. overloaded to_int() to accept 'buffer'.  E.g. buffer b; int x;
  b = read_buffer(file, 0, 4); x = to_int(b);  Note that to_int() expects 
  the integer embedded in the buffer to be in network-byte-order.
 
. In addition to 'int write_buffer(string file, int start, mixed source)'
  there is also 'int write_buffer(buffer target, int start, mixed source)'
  Note that write_buffer(string file, int start, int x) will write 'x'
  into the file (or buffer) in network-byte-order.
 
. In addition to 'buffer read_buffer(string file, int start, int len)'
  there is also 'string read_buffer(buffer b, int start, int len)' which
  provides a way to extract part or all of a buffer into a string.
 
. added two malloc debugging efuns that provide an interface to the malloc 
  debugging system calls provided by the NeXTSTEP OS (malloc_debug() and 
  malloc_check()).  See NEXT_MALLOC_DEBUG in options.h for more info.  
  These efuns aren't available on other OSes.
 
. (: "string" :) is now equivalent to (: this_object(), "string" :)
 
. added replace_program() efun, modified from Amylaar's 3.2.  
 
. added optional argument to query_heart_beat(), object to query the 
  heart_beat of 
 
. changed set_eval_limit() so an argument of 0 resets the current eval_cost, 
  rather than setting the eval limit to 0
 
. added support to initialize local variables.  Any valid expression
  allowed (int x = <exp>;), as with global initializations 
 
. removed several no-longer-used options from options.h (some of the more
  notable ones include support for driver-level host access restrictions
  and shout logging)
 
. changed more efuns so that removing their line in func_spec.c causes most, 
  if not all, of their code to not be compiled in to the driver
 
. removed the following items from the driver config file: "access log file",
  "access allow file", "config directory", "allowed ed cmds" (an up-to-date
  Config.example is in the src/ dir)
 
. the simul_efun object can now be updated and new simul_efuns will go into
  place without having to reboot the driver
 
. hundreds of minor bug fixes and optimizations; see src/ChangeLog for a full
  list
  
0.9.17 to 0.9.18
 
. COMPAT BUSTER: preprocessor: MUDOS_PORT changed to __PORT__, MUDOS_ARCH
  changed to __ARCH__, MUDOS_VERSION to __VERSION__.  Added __COMPILER__,
  __COMPILER_VERSION__, and __OPTIMIZATION__ defines.  Removed the NO_SHADOWS
  define, and added the following possible defines: HAS_SOCKETS, 
  HAS_SHADOWS, HAS_DEBUGMALLOC, HAS_MATH, HAS_MATRIX, HAS_ED, HAS_PRINTF,
  HAS_PRIVS, HAS_EACH, HAS_CACHE_STATS, HAS_RUSAGE, HAS_DEBUG_LEVEL, 
  HAS_OPCPROF.
 
. COMPAT BUSTER: rusage() efun: renamed the "usertime" field to "utime"
  for consistency, and changed it so that the utime and stime fields are 
  always returned in units of milliseconds.
 
. COMPAT BUSTER: removed uid checks from add_worth() efun: if you don't
  want people calling this function, you will need a simul_efun override.
 
. call_other() on an array of objects now returns an array of the return
  values.
 
. Changed printf() to be caught by catch_tell() (if defined in options.h)
 
. Changed notify_fail() to always return 0, allowing for simpler code,
  ie: return notify_fail("Read what?\n");, etc
 
. (s)printf()/sscanf() improvements/fixes.
 
. New efuns: match_path(), query_heart_beat(), origin(), function_profile(),
  strsrch(), errorp().
 
. Added the capability of saving loaded LPC programs to disk (.b files) so
  subsequent loading is faster (added new "save binaries directory" line in
  the runtime configuration file).
 
. Changed dump_prog() to take an optional third argument: the file to
  dump to.
 
. Disk swapping improved: line number info is swapped out, and swapfile
  space is reused.
 
. Added #elif (else if) to the preprocessor.
 
. LOG_CATCHES option added to options.h; if defined, errors caught by
  catch() will still be logged to debug.log, prefixed by "caught: ".
 
. Incorporated Wildcard@TMI-2's (John Fehr: umfehr06@ccu.umanitoba.ca)
  port of MudOS to the Amiga (running AmigaDOS).  The port was made using the
  LATTICE C Compiler.  For those Amiga owners without the LATTICE C compiler 
  (or the desire to compile the driver from the source), the Amiga MudOS 
  binaries are available from actlab.rtf.utexas.edu in 
  /MUD/LPmud/MudOS/amiga.  This port of MudOS does not fully support the 
  socket efunctions.  Any Amiga guru wanting to make them work please feel 
  free (and send your patches to mudos-bugs@actlab.rtf.utexas.edu!).
 
. Added @@ENDMARKER processing, which works similarly to @ENDMARKER, but
  produces an array of strings (1 string per line in the block).  
 
. New second argument to each() efun that resets the current position in the
  mapping kept track of by each().  each(map, 1) will reset the position.
 
. Changed objects() to take two optional parameters: objects(func, ob)
  will pass each object as an argument to ob->func(), and will only return
  the ones for which that function returns 1.
 
. Literally hundreds of bug fixes and optimizations--for full details,
  read the ChangeLog in the ./src directory.
 
0.9.16 to 0.9.17
 
. MudOS now compiles and runs on the DEC Alpha running OSF/1
 
. prelang.y and postlang.y combined into compiler.y - the changes in this
  area should allow the .y files to compile with many more yacc's than
  before (Ultrix, Sun?)
 
. addr_server made more robust and compilable on more machines
 
. Added an OLD_ULTRIX define in port.h for some VAXes running old versions
  of Ultrix; should allow the driver to compile/run on them
 
. Anytime an object is compiled now, valid_object(ob) is called in master.
  If valid_object _exists_, and returns 0, then ob is destructed before
  create() is called and the efun that caused the load errors.  (usage
  of this, along with inherits() and 'nomask', could allow usage of things 
  such as destruct() to be safely restricted through simul_efuns and
  valid_override() in master)
 
. Basic optimization of numeric constants in LPC code is now done (if
  LPC_OPTIMIZE is defined in options.h)
 
. call_other() can now take an array as a second argument, where the first 
  item of the array is the function to call, and all following items will be 
  passed as arguments in the order which they occur
 
. Added an LPC object disassembler (dump_prog() efun)
 
. New resolve() efun which can translate ip names to numbers and vice-versa
  (see efun docs)
 
. Snoops will be sent to receive_snoop(msg) in the receiver object now if
  RECEIVE_SNOOP is defined in options.h
 
. Extension to set_heart_beat(x), where if x > 1, x is used as the number
  of heart beats inbetween heart_beat() calls (can be disabled in options.h)
 
. Extension (3rd argument) to member_array(), which, if set to 1 and using
  a string as 1st arg, checks only the prefix of the strings in the array 
  (similar to xverbs vs verbs)
 
. Added a new efun inherits(string filename, object base) which returns 1
  if base inherits the latest copy of filename, 2 if it inherits an old copy,
  and 0 if it does not inherit filename.  This efun avoids the cost of
  creating an array via deep_inherit()
 
. Added OLD_COMMAND to options.h - if defined, the old second argument to 
  command() will again be available
 
. Added query_shadowing(), a preferred synonym for the shadowp() efun
 
. replace_string() efun expanded to support an optional 4th and 5th 
  argument (see efun docs)
 
. Added query_snooping() efun (inverse of query_snoop()) - might want to
  add a simul_efun override for this
 
. Added query_privs(), set_privs() efuns and privs_file() apply in master 
  if PRIVS is defined in options.h - read the comments there for information
 
. Added a new "function" called time_expression() that returns the number
  of microseconds realtime that occurred during execution of the expression
  passed to it

. Added rusage() support for HP-UX 8.x/9.x (if using 7.x, define OLD_HPUX
  in port.h) and SunOS 5
 
. The writing of text (prefixed with ']') to stderr is no longer done by
  default; can be enabled in options.h (NONINTERACTIVE_STDERR_WRITE)
 
. save_object() now inserts a "#filename" at the top of savefiles 
  (restore_object() ignores such lines; this is useful for virtual object
  compilers that want to figure out how to compile .o files)
 
. functionp(funp) now returns 0 if funp[0] is a destructed object
 
. Driver now calls crash() in master.c with "Host machine shutting down"
  when SIGUSR1 is received
 
. Caught errors (via catch()) are now written to debug.log, prefixed with
  "caught:"
 
. object::func_name() is now allowed (previously, the existence of the 
  'object' type caused conflicts when trying to use object::)
 
. Compat buster: Fixed a bug in multiple inheritance semantics.
   object a: bingme() { bing(); } bing() { write("bingAAA\n"); }
   object b: bingme() { bing(); } bing() { write("bingBBB\n"); }
   object c: inherit a; inherit b;
             bing() { write("bingCCC\n"); }
             create() { a::bingme(); b::bingme() }
  The old behavior would have been to write bingCCC then bingAAA.
  The new behavior is to write bingCCC then bingCCC.
 
. Compat buster: removed arch(), version(), and mud_name() functions.  Added
  MUDOS_ARCH, MUDOS_VERSION, and MUD_NAME defines that are exported to all
  LPC objects - also, the DIR define was changed to MUDOS_DIR
 
. Plenty of bug fixes and optimizations

0.9.15 to 0.9.16                                   
 
. Brought the documentation up-to-date.
 
. Added support for a new basic type: 'float'.  32 bit floating point
  (decimal) numbers, and some basic efuns to deal with them: to_float(),
  to_int(), and floatp().
 
. Added some math efuns for use with floats: cos(), sin(), tan(), asin(),
  acos(), atan(), sqrt(), log(), pow(), exp(), floor(), ceil().  Can be 
  removed by undef'ing MATH in options.h.
 
. Added some efuns for 3D graphics: id_matrix(), translate(), scale(), 
  rotate_x(), rotate_y(), rotate_z(), lookat_rotate(), lookat_rotate2().  Can
  be removed by undef'ing MATRIX in options.h.
 
. New efuns: shadowp(), set_reset(), reclaim_objects(), objects(), 
  memory_info(), reload_object()
 
. Old efun, back again: virtualp()
 
. COMPAT BUSTER: query_snoop() can now be called from anywhere, so you might
  want to put in a simul_efun override for it.
 
. Added a port for HP Motorola (680x0) systems running BSD 4.3.
 
. Added a port for SPARC Classic running Solaris (SunOS 5.1).
 
. Partial ports for Cray (UNICOS 6.1.7) and DEC Alpha (OSF/1 1.2).  Neither 
  is complete as of yet.
 
. Added support for type 'function' in (s)printf()'s %O format spec
 
. SAVE_EXTENSION define is now imported to all LPC code (ie ".o")
 
. Modified save_object() so the original save file is not lost if the save 
  fails due to disk full, quota exceeded, etc.
 
. Added support for comments in object save files (ie .o files).
 
. #ifdef'ed all of the efuns so removing an efun def from func_spec.c
  automagically removes some of the associated code for it.
 
. Continued optimizing many things.
 
. Many more bug fixes (see src/ChangeLog for full information).

0.9.14 to 0.9.15

. added a new feature similar to the << EOF text reading feature
  provided by Perl.  The @ character now means interpret from 'here'
  til the end of the line as the "terminator".  Lex then considers each
  following line up to the terminator as forming part of a string literal.
  An example usage is as follows:
  /* assume this is at the left margin */
  x = @END
  This is taken literally.
  And so is this.
  END
  ;

. optimized the code produced by the LPC loop control constructs (for and
  while). These optimizations are modeled after those appearing
  in Amylaar's enhanced LPmud driver.  MudOS 'for loops' now contain three
  fewer instructions per loop (the empty loop goes from 9 to 6 instructions
  The execution speed of the MudOS empty for loop (not that this is
  that meaningful :) is now within 1% of the speed of the empty for loop
  in Amylaar's driver.

. optimized a few of the LPC operators to produce smaller code.

. added the moncontrol() efun which lets profiling (of the driver - an
  in gmon.out and gprof) be turned on and off from inside the mudlib.

. added support for several new platforms (machines and operating systems)
  including SGI, Linux, 386bsd, A/UX (Apple UNIX(tm)), and HP Apollo.

. fixed the general compile time switch case problem

. added an error() efun which is like a throw() that doesn't require a catch.

. incorporated Amylaar's latest enhanced version of smalloc (uses fast fit).

. added option to options.h to allow save files to have an extension other
  than .o (for those having sysadmins who run scripts that remove .o files
  on a periodic basis).

. changed save_object() and restore_object() so that variables with a value
  of zero (0) needn't be saved to the .o file (however, the old behavior
  is still available for those who want it).  This can produce a noticeable
  savings in disk space.

0.9.0 to 0.9.14

. added a new data type named 'function'.  Be sure to change the names of
  any variables you may have named 'function'.  This data type is a variant
  on the idea of function pointers.  It may be used to store an
  (object, function) pair in a single variable.  The constructor for function
  variables is as follows: 

  function f;

  f = (: obj, func :);

  where obj and func may be anything allowed by call_other(obj, func).
  To call the function func in object obj via function variable f, type:

  (*f)(arg0, arg1, ...); 

  You may use the new functionp() efun to detect if a variable is of type
  function.

. the driver now compiles under the xlc compiler on the RS/6000 (xlc is IBM's
  fastest of AIX 3.2 C compilers).

. added Mappings and Interactives fields to the stats output by mud_status().

. added a new efun set_eval_limit (#define SET_EVAL_LIMIT on options.h) that
  allows an object to change the max eval cost limit.  Be sure to add
  appropriate mudlib level security before enabling this efun (read the
  ChangeLog 0.9.13.4).

. added Cynosure's livings() efun that returns an array of all objects that
  have called enable_commands() (and haven't yet called disable_commands()).

. changed the behavior of nullp() and undefinedp() slightly (see 0.9.13.2
  ChangeLog).

. The catch() efun may no longer be used to trap max eval_cost exceeded
  errors (and thus hang the driver).

. System V Release 4 support (added M.B. Bundy's patches)

. Ridge (RISC/os) support added.

. added capability to configure the default 'What?' message via the runtime
  config file.

. dramatically improved global apply_low (call_other) cache hit rates.  Many
  muds that used to have hit rates near 40% are reporting 93% and above.
  [This improvement results from fixing a bug we found in the LPmud 3.1.2
   cache code.  See ChangeLog for more details.]

. 0.9.14 mappings use about 50% less memory overhead per mapping (compared
  to 0.9.0).

. improved performance.  MudOS 0.9.14 is much faster than MudOS 0.9.0
  at executing many of the more commonly used eoperators.

. created an efuns.h file that should be #include'd from any file that wants
  to define new efunctions (see efunctions.c for example).

. added strcmp() efun (just like C's strcmp) that is convenient for use in
  the compare functions used by sort_array().  sort_array() has been recoded
  to use a robust version of qsort so that ill-behaved compare functions won't
  crash the driver.

. added efuns clonedp() (detects if an object was created via clone_object),
  functionp() (detects if a variable is of type 'function').

. added a new eoperator F_CALL_EXTRA that allows for 512 efuns+eopers instead
  of the previous limit of 256 (easily extended to allow more).  This replaces
  and enhances the F_CALL_EFUN that we had added to the previous release.

0.8.1 to 0.9.0

. Literally hundreds of bug fixes (read the ChangeLog file in the source
  directory for complete descriptions of these if you're interested)

. MudOS now runs on more architectures.  See the Platforms file.

. (internal change) totally rewrote and replaced communication code in the
  driver (comm.c) to be more efficient.

. added socket efunctions for inter-object and inter-mud communication
  (see separate socket efun tutorial and man pages for more a detailed
  explanation)
        * int socket_create(int, string, string|void);
        * int socket_bind(int, int);
        * int socket_listen(int, string);
        * int socket_accept(int, string, string);
        * int socket_connect(int, string, string, string);
        * int socket_write(int, mixed, string|void);
        * int socket_close(int);
        * int socket_release(int, object, string);
        * int socket_acquire(int, string, string, string);
        * string socket_error(int);
        * string socket_address(int);
        * void dump_socket_status(void);
        * void dump_file_descriptors();

. added MUD mode to the LPC socket efunctions.  MUD mode allows LPC
  data structures such as arrays, mappings, integers, and strings (all
  except for objects) to be sent atomically via sockets.

. added DATAGRAM (UDP) mode to the LPC socket efunctions.

. (internal change) rearchitected interpret.c to use a function lookup table
  rather than one large switch statement.  New efunctions should be added
  to efunctions.c instead of interpret.c.  This enhancement makes it
  easier to integrate your own efuns into the driver and it allows interpret.c
  to compile on machines that previously had problems with it.

. added call to master.c::valid_override(string filename, string function)
  for each attempt to override a simulated efunction via the
  efun::func_name() method.  'filename' is the path to the file which
  is attempting to call efun::func_name().  'function' is the name of
  the efun that is to be overridden.  If valid_override returns 0, then
  the current compilation will fail (note: valid_override is only called
  at compile time and _not_ runtime).  valid_override allows
  valid_destruct, valid_exec, etc. to be dispensed with since the same
  functionality can be obtained by wrapping the efun in question with a
  simul_efun of the same name (and including valid_* style checking in the
  simul_efun).  Gist of the idea taken from Amylaar's done file.

. added the option of configuring the driver so that resets (the calling of
  reset() in LPC objects) happen in a lazy fashion (by defining LAZY_RESETS
  in options.h).  This option allows resets to happen in an incrementally
  (rather than all in one pass) and it prevents resets from occuring unless
  the object to be reset is touched (via an apply_low or move_object) after
  the usual required reset delay has occurred.  The benefits of this option
  are that performance is smoother (no reset lags) and memory is conserved
  (since those object which are touched once and never again won't have reset
  called in them and hence won't reload any objects in reset()).

. changed various efunctions to honor the hidden object flags so that
  the set_hide() efunction now offers true driver-supported invisibility
  of objects.

. new internal implementation of mappings (associative arrays) which is
  faster and more robust.

. new efuns for mappings:
        * keys() - this used to be named indices()
        * values() - the inverse of keys()
        * each() - traverse a mapping

. split config.h into options.h and port.h.  This simplifies the task
  of configuring the driver to run on a particular machine.  In most
  cases, it will be sufficient to configure options.h (options.h contains
  configuration options pertaining to preference rather than portability).

. fixed make_func.y so that it correctly processes func_spec.c on the
  HP and Sequent.

. added a NO_ANSI option to options.h that allows escape codes included
  in user commands to be replaced with spaces.

. added a MAX_SAVE_SVALUE_DEPTH macro to options.h which solves the problem
  of calling save_object on an object containing a data structure with
  a circular (recursive) reference.  Admittedly, this isn't a true fix since
  the object in question (having the circular reference) cannot be saved;
  however, this is better than allowing the driver to hang.  (Yes I know
  that Amylaar solved this by saving the circular reference in a recoverable
  fashion.  I didn't consider it an important enough problem to spend the
  time necessary to adopt his solution).

. added a new efunction replace_string() (see the replace_string man page)

. added a clonep() efunction that determines if an object was created
  via the clone_object() efunction.

. added a localtime() efunction (see the localtime(3) man page)

. added an uptime() efunction that returns the number of seconds since
  the driver has been running.

. removed the mudwho code from the driver (including clilib.c).  We have
  implemented the mudwho protocol (same as that implemented in clilib.c)
  in LPC using the new MudOS socket efunctions (DATAGRAM mode).  See the
  README in the included mini-mudlib for more details.

. changed inherit_list() and deep_inherit_list() not to return the filename
  which is passed to them.

. added a configuration option that causes this_player() to return 0 inside
  callbacks made by call_out().

. heartbeat interval can be now be specified in units of microseconds on
  those machines with the ualarm() system call.

. size of the call_other cache table is configurable (via the
  APPLY_CACHE_BITS define in options.h).* (internal change)
  totally rewrote and replaced communication code in the driver (comm.c) to
  be more efficient.

. made the ed() efunction (internal editor) optional.

. input_to can now take extra arguments so that you can pass one or more
  extra arguments to the function input_to calls (notice
  that this feature makes it possible to avoid using global variables which
  makes it easier for a single object to be designed to
  beshared by many players [e.g. daemon-style objects]).  For example:

        void get_name(string name, object new_player)
        {
                write("\nWhat is your email address: ");
                input_to("get_email", 0, new_player, name);
        }
        void get_email(string email, object new_player, string name)
        {
                new_player->set_email(email);
                new_player->set_name(name);
        }

. implemented variable arguments to call_out just like the additions made to
  input_to described above.

. You can now compare mappings for equality using the == operator.

. the message() efun now works. For more info read the accompanying doc on
  this important new efun!

. several new required functions to be implemented in the mudlib master.c:
        * valid_socket - security to monitor or limit usage of the new socket efuns
        * valid_domain - a check for domain_stats to make sure that it only keeps
          stats on domains that you care about
        * valid_hide - for the set_hide() efun so that who can be "hidden" is
          controlled by master.c
        * slow_shutdown - when the driver is running out of memory and wants a
          nice shutdown it calls this

. several new additions to the runtime configuration file:
        * inherit chain size : the maximum inheritance depth allowed.  This is
          mainly to prevent infinite inheritance loops
        * maximum mapping size : similar to maximum array size
        * address server ip : machine name where an addr_server is running
        * address server port : port number that the addr_server is listening on
        * max users : this used to be max players (doesn't matter anyway, since
          this is still unused by the driver here)
        * maximum efun sockets : control the number of efun sockets open at once
        * maximum string length: controls the maximum allowed string size

. wizlist() changed to domain_stats(), this is to make statistic taking more
  consistant with the new domain model versus the wizard model previous
  to LPmud 3.0.  See also: author_stats()

. valid clone - this isn't enabled by default, but with some changes to
  options.h, clone_object will fail if an object doesn't have read permission
  on the file being cloned. (note: this is pretty useless since inheritance
  doesn't have a similar check)

. old "parser magic functions" were changed into efuns:
        * malloc_status() - was the 'malloc' command
        * dumpallobj() - was the command of the same name
        * mud_status() - was 'status' (use mud_status(1) to get 'status tables'

. some new debugging efuns were added or changed:
        * rusage() now returns a mapping of the values returned by the
          getrusage() system call.
        * debugmalloc() - if the driver is compiled with malloc debugging enabled
          this prints out more detailed malloc information
        * set_malloc_mask()  - control the level of info being debugged with
          debugmalloc
        * set_debug_level() - more control on debugging levels (see debug.h in
          the src directory)
        * opcprof() - dump a file detailing efun usage (usage frequencies)
        * cache_stats() - display call_other function table cache hit rate
          information

. removed the old hname process and replaced it with a new autonomous (not
  forked from driver) addr_server program that can be shared by several
  different drivers at once.

. removed the cindent() efun

. removed the describe() efun

. removed two unused variables from the runtime config system
        * global include file
        * max commands per heartbeat

. removed everything to do with the "priveleged object" concept
        * set_priv()
        * privp()

. removed everything to do with "global verbs", "global preverbs" and
  "preverbs"

. all objects now call create when first loaded, even if loaded via
  inheritance (create called in the original objects, not what's
  being inherited)

. crash in master.c is now called with three args (type of crash,
  command_giver, current_object)

. if the driver crashes it tries to dump its core file in <mudlib dir>/cores
  if that dir exists and core dumping is enabled

. removed the global define LPCA and replaced it with MUDOS (LPC objects can
  do #ifdef MUDOS)

. added AUTO_SETEUID option.  if defined, all objects do the equivalent of
  seteuid(getuid(this_object())); when created.

. added AUTO_TRUST_BACKBONE define so that you could disable the automatic
  trusting of backbone uid objects if you want.

. added Pinkfish's modifications to parse command package so that it 
  actually works now. :-)

. added an optional integer argument to the shutdown() efun.  when an
  argument is specified, the driver exits with that exit code.

. removed smalloc from the driver completely (memory alignment violations and
  other problems made it not worth dealing with).

. added BSD malloc as an option (seems to work well for those machines having
  slow system mallocs like the HP).

. added the concepts of "null" and "undefined" to variables and added two
  efuns to make use of these concepts
        * nullp() - returns true for unitiliazed variables and destructed objects
        * undefinedp() - returns true for (1) call_others to non-existant
          methods, (2) access to map keys that aren't in the map, and
          (3) arguments in the formal parameter list of a function for which
          values were not passed.

. there's a DROP_CORE option in options.h which allows you to specify if you
  want the driver to try to dump core if it has crashed

. replaced sort_array() with a faster version.  Also changed it so that
  it expects the callback function to return -1, 0, or 1 in a fashion
  similar to how strcmp() does it.

. added an option to define the default error message users receive.  If it
  is not defined, then all users get wizard level error messages.  This is
  useful since players are the ones that most often experience these errors
  and this lets players report them in a more coherent fashion (maybe?).

. Reworked the domain stuff (added uid.c and uid.h) by separating the
  concept of uids and euids from that of domains and authors.  The
  wizlist (domain list) code is no longer based on uids but rather on
  domains.  Two new functions are required in master.c:
  string domain_file(string domain); and string author_file(string author);
  In addition, valid_author(string author) should be added to master if
  you wish set_author(string author) to be functional.  Note: domain_file
  and author_file are similar to creator_file (which returns information
  for uids).  domain_file lets the gamedriver know which domain a given
  object should be in an author_file lets the gamedriver know the author
  of the file.

up to 0.8.1

. COMPAT_MODE is gone!  You will not be able to run in compatibility mode.
  This means that MudOS cannot be used to run mudlibs designed for the
  2.4.5 LPmud gamedriver.

. shadow() can be disabled at compile-time.  Check out NO_SHADOWS in
  config.h.  Note that NO_SHADOWS will appear in LPC's preprocessor's
  run-time table if it is defined in config.h.
   
. A runtime configuration file replaces much of config.h.  Look at the
  file ./mudlib/config.example.  You must specify a config file as
  an argument to the driver when starting.  It will not run without it.
  The driver will search in CONFIG_FILE_DIR (defined in config.h) for
  the file first and will then search the explicit path.
 
. Added Sean Reith's sprintf().  It's huge and therefore a compile-time 
  option defined in config.h.

. Added RCS as a compile-time option.  This allows wizards access to
  the new efuns ci, co and rlog if those executables exist on your
  system.

. added dprint flag for .edrc file.  Dprint causes a line to be printed
  following the deletion of some text in ed.

. Compatibility buster: file permission system is now object-based
  instead of euid-based.

. save_object() no longer saves 0-valued variables.

. Virtual objects.  When the GD is asked to compile a file that
  doesn't exist, it just forwards the request to compile the file name
  to the master object.  The return value of compile_object(fname) will
  be an object that will be renamed.

. set_prompt() from interactive object sets that object's prompt to the
  string argument.

. If the function write_prompt() exists, it will get called every time
  the driver would normally write a prompt.  (this only gets called when
  not in ed or an input_to)

. Added privileged object concept.  The master object is by default
  privileged; enable_privileges(ob) from a privileged object makes ob
  privileged; privp(ob) returns whether or not ob is privileged.

. wizards() returns a string array of all the wizards in the wizlist.
  find_wizard(wizname) returns information about that wizard, in an
  array.  The information returned is: moves, eval cost, errors, heart
  beats, "worth", size of arrays, and number of objects.  This is the
  information since the last reboot.  It is guaranteed to be in this
  order (more fields'll be added later).  If the wizard is not found, 0
  is returned.

. Crash vector added.  When the game driver receives some signal such
  as segfault, bus error, etc., crash() is called in the master object
  with the string error that was generated.

. Symbol LPCA is #defined by preprocessor.

. input_to()'s second argument is now a bitmask number.  0 and 1 work as
  before; 2 doesn't allow shelling out; 3 is like 1 and doesn't allow
  shelling out.  This applies to compatibility mode too, unlike
  receive().  Be sure to use these flags at login time...  localcmd()
  has been removed, check out commands().


. Only objects with same user id as master object (root uid) or backbone 
  uid, are allowed to add_worth().

. Gamedriver now keeps track of mappings on status line.  Fixed a memory leak 
  in mapping aggregate initialization.  Mappings now appear in the arrays 
  section of the wizlist.

. Preprocessor now supports token pasting.  To paste two tokens in a #define, 
  use '##'; e.g. #define SET(str, v) set_##str(v)

. inherit statement now understands string expressions.

. You can now clone an object with an environment.

. Privileged objects can't be destruct()'d by non privileged objects.

. Snooping of ed mode looks prettier, also improved substitution.

. New data type added: mapping.  Usage:
      declaration-     mapping m;
      initialization-  m = (["fighting" : 35, "swimming" : 3]);
      access-          if (num_users["netcom"] > 3) ...
      modify-          skills[skill_name] += 20;

   Mappings are like arrays but you can index off of strings,
      numbers (even negative ones), arrays, objects, and
      mappings.  save_object() and restore_object() seem to
      handle them fine.

   * works with mappings, as a composition operator.  Likewise '*='.

   Alist efuns are no longer.  Alists have been obsolesced by mappings.

. get_dir() now takes second optional flag argument.  Currently, the only
  flag understood is -1.  If this flag is chosen, instead of returning
  an array of strings get_dir() returns an array of arrays which are
  information about the file.  Information is returned in the order:
  filename, filesize, file update time.
  New efun stat() acts just like get_dir(), and will replace it.

. Swap file name now includes port number, so that it's possible to run
  debug mud off of same mudlib without messing things up.

. Third argument to add_action() is now a bitmask.  Flags:
      1 - short verb
      2 - xverb
      4 - preverb (new)
      16 - global verb (new)
      Global verbs can only be added from privileged objects.
      Preverbs can only be added from privileged objects or
      the command giver.

  Parse order:  global preverbs, then local preverbs, then
                local verbs, then global verbs.

  The "n", "s", "e", "w" hardcoded aliases have been removed
     from the GD, you'll have to use global verbs to do the
     same thing.
 
. Added new modifier to sscanf(): '*'.  It allows you to skip over a
  field without having to use a dummy variable.  e.g.
    sscanf(file_name(ob), "%*s#%d", num);
  It's just an efficiency hack; you don't really need this.
   
. C++ style comments (//) now work.

. Restricted ed mode.  If this option was compiled in, if the
  interactive object which called ed isn't a wizard (wizardp() == 0),
  that object is in "restricted ed mode".  This only allows the user to
  write to the file that ed was started up in ('x' is the only save
  command which is allowed).  In addition, help messages etc. are
  tailored to the limited set of commands.

. Netdeath notification.  Upon going netdead, the object which goes
  netdead has the function "net_dead" called on it.

. cp is back in the parser as an efun.

. renamed inherit_list efun to deep_inherit_list (since this efun does
  return ALL of the files inherited by the object in question)
  and made a new efun called inherit_list that only returns a list of
  the files directly inherited by the object in question.  This efun
  will be useful when writing a deep update command.

. overloaded the definition for the stat() efun so that if it is applied to a
  regular file instead of a directory, it returns information on
  that file.  The information returned is a three element array:
  (file_size, file_modification_time, update_time_of_associated object)
  The times returned will be useful in writing a deep update command.
  This overloaded version of stat requires a full pathname as an argument.
  You can check if stat() fails by seeing if the returned vector has size 0.
  (e.g. if (sizeof(v) == 0) handle_error())

. Overloaded call_other() so that it can take an array as the first
  argument.  e.g. users()->quit(); (the return value is always 0 now)

. move_object(ob) is now equivalent to move_object(this_object(), ob).
  This may phase out the latter usage.

. Every object now has the macro DIR defined, which is the directory 
  that the object's .c file is in.

. process_input() is called on the player object if it exists for
  every line of input passed to the parser by that player.
  process_input should return a string.  That string is the actual
  string which is parsed by the driver.  This allows you to do things
  like nicknames or aliases quickly and easily.

. Made in_edit and in_input take an optional argument which is an
  object, so you can check the status of other objects.


. New efuns:
   userp(ob) returns whether or not ob is a user.

   new(obname) returns clone_object(obname).  clone_object()
      won't be around for long.

   remove_action(func, verb) added.

   master() returns the master object.

   in_input() returns 1 if current object is in an input_to().

   in_edit() returns 1 if current object is editing.

   mapp() returns 1 if argument is a mapping.

   receive(str) sends interactive message to current object,
      returns 1 if current object is interactive.

. commands() returns an array of arrays detailing the sentences of the
  current object.  Subarray structure: 0=verb, 1=flags,2=defining
  object.
 
. enable_wizard() makes current object into a wizard.  Wizards don't
  add moves on the wizlist, and get better error reporting than "your
  overly sensitive mind...".

. wizardp() returns whether argument is a wizard or not.

. ci, co, rlog (if the parser was compiled with RCS turned on).
  ci (string filename, string log_messag, int major, int minor); 
  co (string filenamem, int major, int minor);
  (major and minor are the major and minor version numbers)
  rlog (string filename);

. get_char().  Works exactly like input_to, except that it responds to
  a single character of input.  Warning: this doesn't work well with
  clients.

. mud_name() Returns the name of the mud as defined in the config file.

. allocate_mapping(int) 
  This efun isn't strictly
  required in order to use a mapping, it merely serves to give the gamedriver
  a hint regarding how many elements the mapping may be expected to
  eventually contain.  So long as the mapping contains fewer than the
  specified maximum number of elements, the gamedriver will use
  contiguous memory to store the elements.  Once this number of elements
  is exceeded, the game driver begins malloc'ing the space for the
  elements separately as needed.  Note that using "map = ([])" to
  initialize the map is the same as using "map = allocate_mapping(0)".

. set_hide(int) allows a privileged object to hide itself from
  find_living(), find_object(), find_player().

. map_delete(mapping,index) - removes a given pair (index, value) from
  the mapping 'mapping'.

. keys(mapping) - returns an array of the indices of a mapping.

. undefinedp(mixed) - returns TRUE if the argument is undefined.  You can
  check for inclusion of a given index in a mapping with undefinedp.  E.g.
  if (undefinedp(x["index"])) write("index isn't in mapping 'x'\n");
  This function also works on values returned from call_other.  Call_other's
  on non-existant functions return values that cause undefinedp to evaluate
  to TRUE.

. children(object) - returns an array of all the objects which were cloned
  from the argument passed to it.

. message(string,string,mixed,object *) - general purpose replacement
  This efun is a generic messaging function meant to replace all current
  communication efuns in the future (say, shout, tell_*, printf,
  write, etc).  It communicates directly to catch_tell in the
  targets.

. Virtual objects
   The immediately obvious thing to use them for is virtual
   territories.  Want to create a desert with 10,000 rooms?  You
   can, now.  You could even make it a 'sparse territory' by making
   it mostly virtual, with some occasional detailed rooms thrown in.
   The recommended strategy is to have the compile_object() code
   in the master object look like:
   
    object compile_object(string fname) {
       return (object)
          "/secure/virtual_object"->compile_object(fname);
    }
 
   This keeps the complexity of the master object down.  Then
   have the virtual_object server delegate its responsibility.  e.g.
   if an undefined room occurs in a wizard's castle, have it call
   compile_object(fname) in that wizard's castle.c file.  Better do
   a security check as well.