home *** CD-ROM | disk | FTP | other *** search
- //----------------------------------------------------------------------
- // MegaJitter V1.3 written by L. Vanhelsuwé (C) LVA 1992-94
- // ---------- ---- ------------------------ ---------------
- // Based on a slower program called "Jitter" (authors "Don & Chris").
- //
- // Graphical evolution simulator.
- // Allow gene-based creatures to evolve within an environment.
- //
- // Features of this simulation include:
- // - gene-based control over characteristics
- // - mutation of genes at birth
- // - sexual and asexual reproduction
- // - creatures can be 100% herbivores... omnivores... 100% carnivores
- // - real-time display of entire ecosystem
- // - real-time display of statistics
- // - flexible control over ecosystem global variables
- // - AREXX interface
- //
- // History
- // -------
- // XX-AUG-92: Started this file.
- // 12-OCT-93: Added greyscales display option for crappy grey VGA monitor.
- // 02-JAN-94: Added Food vision capability to creatures.
- //
- // VERSION 1.2
- //
- // 13-MAR-94: Added a faster non-graphical (NOANIM) mode (using bytes for cells)
- // + scenario settings now in variables instead of #defs.
- // 26-MAR-94: Changed code to use pointers to functions for CLI selectable
- // non-animated (faster) mode.
- // Added dying_age and moving speed genes. "BLIND" option.
- // 08-APR-94: Added sexuality... at last.
- // 10-APR-94: Started adding MenuStrip with dynamic statistics selection.
- // 12-APR-94: Finished Stats Menus.
- // 13-APR-94: Added Options Menu with Graph speeds as mutually excluding
- // sub-items.
- // 16-APR-94: Added (non satisfactory) attempt to be NTSC-friendly.
- // 17-APR-94: Changed defaults to be NO SEX and NO VISION
- // 22-APR-94: Added checks for out-of-bounds global parameters on CMD line.
- // 27-APR-94: Added herbivore/carnivore capability
- // 01-MAY-94: Added my own AREXX compatibility, after thinking that the PD
- // offerings were too complicated.
- // 03-MAY-94: Added rexxsyslib IsRexxMsg() sanity check for incoming msgs.
- //
- // VERSION 1.3
- //
- // 11-MAY-94: Changed Oasis food growth to true circular area.
- // Killed some REXX bugs, created first REXX scenarios.
- // 12-MAY-94: Major gfx optimization: now using an interleaved bitmap for faster
- // pixel routines.
- // 17-MAY-94: Made MAX_ENERGY a gene instead of a global constant.
- // 19-MAY-94: Forced screen Font to be Topaz 8*8
- //
- // TO DO
- // -----
- // - LOAD/SAVE entire state.
- // - Global parameters control panel
- // - Help menu
- // - long vertical hardware sprite as a crosshair type cursor over graphs.
- //
- // Lines marked **!! are potential bug sources.
- //
- // DESIGN MISTAKES:
- // ----------------
- // - When I added the NOANIM option and therefore the necessary parallel
- // representation, I started a second path of development which had to
- // have the same functionality as the original pixel-based system.
- // This is bad: too much work maintaining the 2 systems.
- // I should have used one single data structure capable of being displayed
- // as pixels or not (as in NOANIM).
- //----------------------------------------------------------------------
-
- #ifdef DEBUG
- #define ENTER(x) printf("Entering..." x "\n");
- #define LEAVE(x) printf("Done ..." x "\n\n");
- #endif
-
- #define MONOCHROME 1 // running on a mono display!
-
- //#define FASTER 1 // enable to cut out IFAUDIT code...
-
- // Major apology to the whole world out there: I am polishing this program on
- // a minimalistic grey-scale VGA monitor, so, sorry about the B&W look !
-
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <math.h>
-
- #ifdef AMIGA
- #include <exec/types.h>
- #include <exec/execbase.h>
- #include <exec/memory.h>
-
- #include <graphics/gfx.h>
-
- #include <intuition/intuitionbase.h>
- #include <intuition/screens.h>
-
- #include <libraries/asl.h>
- #include <libraries/gadtools.h>
-
- #include <utility/tagitem.h>
-
- #include <rexx/rxslib.h> // for AREXX Support !
- #include <rexx/errors.h>
-
- #include <clib/exec_protos.h> // ANSI function prototypes
- #include <clib/alib_protos.h>
- #include <clib/dos_protos.h>
- #include <clib/graphics_protos.h>
- #include <clib/intuition_protos.h>
- #include <clib/gadtools_protos.h>
- #include <clib/rexxsyslib_protos.h>
-
- #define LF 10
-
- #else // if compiled on non-Amiga machines..
-
- typedef unsigned char UBYTE; // 8 bits
- typedef unsigned short UWORD; // 16 bits
- typedef unsigned long ULONG; // 32 bits
-
- #endif
-
- typedef unsigned int FLAG; // 32 bits **!! optimized for 68020+
- typedef unsigned int RC;
-
- //----------------------------------------------
- #define CHAR_HEIGHT 8 // Y size of Font.
-
- #define SCREEN_WIDTH 512 // Screen dimension **!! FIXED
- #define SCREEN_HEIGHT 512 // HAS to be a power of 2.
- #define ACRE (SCREEN_WIDTH-1) // POWER OF 2 **!! LEAVE AS IS
- #define ECO_SCR_DEPTH 4 // for 16 colors
-
- #define STATS_TOPY 12 // offset from top of screen.
- #define GRAPH_HEIGHT 64
- #define STAT_SCR_WIDTH 640 // Statistics Screen Width
- #define STAT_SCR_DEPTH 3 // only 8 colors needed.
-
- #define NUM_COLORS (1<<ECO_SCR_DEPTH)
- #define NUM_CELLS (1<<ECO_SCR_DEPTH)
-
- #define NEUT_CELL 0 // has to be all 0s eg. %0000
- #define VEGE_CELL (NUM_CELLS-1) // has to be all 1s eg. %1111
- #define WALL_CELL (NUM_CELLS-2) // has to be all 1s eg. %1110
- #define MAX_COLORS (NUM_CELLS-3)
-
- #define SYNTAX_ERROR (5)
- #define FATAL_ERROR (20)
- #define PROG_BUG_ERROR (100)
-
- #define REXXERR_UNKNOWN_CMD (10)
- #define REXXERR_VAL_OUTSIDE_RANGE (11)
-
- //----------------------------------------------
- // Statistic Graphs constants
- //----------------------------------------------
-
- #define TIMER_SPEED 4 // Time update speed
- #define LE (STAT_SCR_WIDTH-256) // LABEL area for graphs
-
- #define NORMAL 0
- #define FLIP_IT 1 // don't change **!!
-
- #define CURGRAPH_Y (STATS_TOPY+ graphnum*GRAPH_HEIGHT) // current graph Y BASE
-
- //-----------------------------
- // Main program data structures
- //-----------------------------
-
- ULONG time; // And God Createth Universal Time
- UWORD runs=0;
- ULONG daylight; // +100 to -100 (noon, midnight)
-
- struct fast_pixel {
- ULONG pixel_byte_offs; // offset of byte in plane
- ULONG pixel_bit_no; // minimal pixel info
- };
-
-
- // The BUG data structure is this program's core data structure, therefore
- // keep field positions optimized for a 32-bit data bus processor.
-
- struct bug {
-
- FLAG alive; // flag enabling all the other fields
-
- // Individual time-dependent creature characteristics
-
- ULONG age; // howmany timer ticks he's been around.
- WORD energy; // 0..procr_energy (MAX +32767 **!!)
- UWORD x,y; // screen coords of bug.
-
- UWORD generation; // which generation this guy is
-
- struct fast_pixel prevpix; // prev pixel info (for fast clear)
-
-
- UBYTE dir; // 0..7
- UBYTE speed_delay; // 0..3
-
- // GENES (time-independent creature characteristics)
-
- UBYTE right; // 0..255
- UBYTE left; //
-
- UBYTE vision; // 0..255 scaled down to ..
- UBYTE sexuality; // 0..255
-
- UBYTE dirgene; //
-
- UBYTE oxy_produce;
- UBYTE oxy_consume;
- UBYTE co2_produce;
- UBYTE co2_consume;
-
- UBYTE dying_age; // 0..255 but scaled up to 0..65535
-
- UBYTE maxsexage; // 0..255 but scaled up to 0..65535
- UBYTE minsexage; // 0..255 but scaled up to 0..4095
-
- UBYTE speed; // 0..255 but scaled down in delay
- UBYTE foodtype; // 0..63 HERBI, 64..191 OMNI, 191..255 CARNI
-
- UBYTE procr_energy; // procreate energy threshold 0..16384
-
- UBYTE dummy;
- UWORD dummyw;
- };
-
- struct bug dummy_bug; // one instance to pass as dummy arg.
-
- typedef UBYTE CELL; // a cell in the 2-d array
- typedef struct bug *BUGPTR;
-
- // array holding MAX_BUGS simulated creatures (allocated at run-time)
-
- BUGPTR bugs; // struct bug bugs[MAX_BUGS];
-
- // To allow creatures to find out efficiently which "bug" structure is
- // connected to a particular creature pixel on-screen, we use a parallel
- // map to hold pointers to the "parent structures".
-
- CELL *environment; // ptr to BYTE-based env of creatures
- BUGPTR *owner_ptrs; // ptr to parallel PARENTS map
- #define PTR_MAP_SIZE (sizeof(char *)*SCREEN_WIDTH*SCREEN_HEIGHT)
-
- int color_divider; // factor to map bug->energy to pixel pen number
-
- //-----------------------------------------------------------
- // The following variables are counters for statistics/graphs
- //-----------------------------------------------------------
-
- ULONG numbugs;
- ULONG food_in_system;
- ULONG births,deaths;
- ULONG total_energy;
- ULONG total_age;
- ULONG total_right, total_left, total_vision, total_dir;
- ULONG total_dying_age, total_minsexage, total_maxsexage;
- ULONG total_speed, total_sexuality;
- ULONG total_foodtype, total_procr;
-
- ULONG total_O2, total_CO2;
-
- UBYTE min_right, max_right;
- UBYTE min_left, max_left;
- UBYTE min_vision, max_vision;
- UBYTE min_dying_age,max_dying_age;
- UBYTE min_speed, max_speed;
- UBYTE min_sexuality,max_sexuality;
- UBYTE min_minsexage,max_minsexage;
- UBYTE min_maxsexage,max_maxsexage;
- UBYTE min_foodtype, max_foodtype;
- UBYTE min_procr, max_procr;
-
- ULONG oxygen,carb_dioxide;
-
- UBYTE stat_bufix=0; // statistics buffer INDEX
-
- struct statistic {
- FLAG displayed; // currently displayed or not
-
- char *label; // Menu and Label string
- ULONG *stat_addr; // ptr to statistic variable
- UBYTE *minmax_addr; // -> minimum and maximum stats (if non-NULL)
- ULONG scale; // maximum value equivalent to max scale
- FLAG flip_graph; // normal or up-side-down graph ?
- ULONG *buffer; // buffer[512] record circular buffer
- };
-
- #define NUM_STATS (19)
-
- struct statistic available_stats[] = {
- { TRUE ,"Number of Bugs....................." ,&numbugs ,NULL ,34 ,NORMAL ,0 },
- { FALSE ,"Number of Bug Births..............." ,&births ,NULL ,64 ,NORMAL ,0 },
- { FALSE ,"Number of Bug Deaths..............." ,&deaths ,NULL ,64 ,NORMAL ,0 },
- { FALSE ,"Average Bug Energy................." ,&total_energy ,NULL ,32768 ,NORMAL ,0 },
- { FALSE ,"Average Bug Age...................." ,&total_age ,NULL ,65535 ,NORMAL ,0 },
- { TRUE ,"Amount of food....................." ,&food_in_system ,NULL ,5000 ,NORMAL ,0 },
- { TRUE ,"Oscillator: Day/Night.............." ,&daylight ,NULL ,100 ,NORMAL ,0 },
- { TRUE ,"Gene Avg.: Clock-wise.............." ,&total_right ,&min_right ,255 ,FLIP_IT ,0 },
- { TRUE ,"Gene Avg.: Anti Clock-wise........." ,&total_left ,&min_left ,255 ,FLIP_IT ,0 },
- { FALSE ,"Gene Avg.: Vision.................." ,&total_vision ,&min_vision ,255 ,NORMAL ,0 },
- { FALSE ,"Gene Avg.: Sexuality..............." ,&total_sexuality ,&min_sexuality ,255 ,NORMAL ,0 },
- { TRUE ,"Gene Avg.: Lifespan................" ,&total_dying_age ,&min_dying_age ,65535 ,NORMAL ,0 },
- { FALSE ,"Gene Avg.: Min. Procreation Age...." ,&total_minsexage ,&min_minsexage ,4096 ,NORMAL ,0 },
- { FALSE ,"Gene Avg.: Min. Procreation Energy." ,&total_procr ,&min_procr ,16384 ,NORMAL ,0 },
- { FALSE ,"Gene Avg.: Max. Procreation Age...." ,&total_maxsexage ,&min_maxsexage ,65535 ,NORMAL ,0 },
- { FALSE ,"Gene Avg.: Moving Slowness........." ,&total_speed ,&min_speed ,255 ,NORMAL ,0 },
- { TRUE ,"Gene Avg.: Herbi-/Omni-/Carnivore.." ,&total_foodtype ,&min_foodtype ,255 ,NORMAL ,0 },
- { FALSE ,"Amount of Oxygen..................." ,&total_O2 ,NULL ,60000 ,NORMAL ,0 },
- { FALSE ,"Amount of Carbon Dioxide..........." ,&total_CO2 ,NULL ,60000 ,NORMAL ,0 },
-
- { 0 ,"**!! MENU BUG: GONE ONE TO FAR *." ,NULL ,NULL ,0 ,0 ,0 }
- };
-
- #define EYE_ITEM (9)
- #define SEX_ITEM (10)
- #define OXY_ITEM ( (sizeof(available_stats)/sizeof(struct statistic)) -3)
- #define CO2_ITEM ( (sizeof(available_stats)/sizeof(struct statistic)) -2)
-
- UWORD graphnum; // current graph index
- UWORD graph_speed; // current graph update speed
- UWORD MAX_GRAPHS; // 6 or 8
- UWORD STAT_WIN_HEIGHT;
-
- FLAG NUMERIC_STATS = TRUE;
- FLAG MINMAX = FALSE; // draw minima/maxima too ?
-
- //-------------------------------------------------------------------------
- // The following AmigaDOS-filled structure is used to store program globals.
- // The structure is initialized by DOS from the options the user passes
- // on the command line.
- // The program itself post-initializes any globals not set by the user.
- //-------------------------------------------------------------------------
-
- struct RDAres { // array for ReadArgs()
-
- ULONG audit; // report events on STDOUT ?
- ULONG capture; // save all stats to file ?
- ULONG noanim; // no animation of simulation ?
- ULONG eyes; // enable vision gene ?
- ULONG sex; // enable sexual pairing ?
- ULONG showdefaults;
- ULONG maxbugs;
-
- // Starting SCENARIO variables
- // ---------------------------
-
- ULONG INIT_BUGS;
- ULONG INIT_FOOD;
- ULONG INIT_ENERGY;
- ULONG INIT_VARIANCE;
- // ULONG INIT_O2;
- // ULONG INIT_CO2;
-
- ULONG MUTATE_RANGE;
- ULONG FOOD_RATE;
- ULONG FOOD_ENERGY;
-
- ULONG X_AND;
- ULONG Y_AND;
-
- ULONG OASIS_SIZE;
-
- ULONG ntsc; // User supplies NTSC/PAL information !
-
- } options;
-
- #define ASEXUAL (FALSE)
- #define SEXUAL (TRUE)
-
- #define REPORT (options.audit)
- #define CAPTURE (options.capture)
- #define NOANIM (options.noanim)
- #define VISION (options.eyes)
- #define SEX (options.sex)
- #define SHOWDEFAULTS (options.showdefaults)
- #define MAX_BUGS (options.maxbugs)
- #define INIT_BUGS (options.INIT_BUGS)
- #define INIT_FOOD (options.INIT_FOOD)
- #define INIT_ENERGY (options.INIT_ENERGY)
- #define INIT_VARIANCE (options.INIT_VARIANCE)
- //#define INIT_O2 (options.INIT_O2)
- //#define INIT_CO2 (options.INIT_CO2)
-
- #define MUTATE_RANGE (options.MUTATE_RANGE)
- #define FOOD_RATE (options.FOOD_RATE)
- #define FOOD_ENERGY (options.FOOD_ENERGY)
- #define X_AND (options.X_AND)
- #define Y_AND (options.Y_AND)
- #define OASIS_SIZE (options.OASIS_SIZE)
- #define NTSC (options.ntsc)
-
- #define IFAUDIT if (REPORT)
-
- #ifdef AMIGA
-
- char MJIT_DOS_TEMPLATE[] =
-
- "AUDIT/S,DATALOG/S,NOANIM/S,VISION/S,SEX/S,SHOWDEFAULTS/S,MAX_BUGS/K,\
- INIT_BUGS/K,INIT_FOOD/K,INIT_ENERGY/K,INIT_VARIANCE/K,MUTATE_RANGE/K,\
- FOOD_RATE/K,FOOD_ENERGY/K,X_AND/K,Y_AND/K,\
- OASIS_SIZE/K,NTSC/S";
- #endif
-
- // The Variable structure defines program globals which can be accessed or
- // changed via the command line and/or the AREXX interface.
-
- struct variable {
- UBYTE flags; // global and/or init
-
- char *name; // name of a global parameter
- ULONG *address; // its address
-
- ULONG init_val; // initial default value
- ULONG min_val; // range minimum
- ULONG max_val; // range maximum
- };
-
- #define PARAMETER 1
-
- struct variable varlist[] = {
- {0 ,"MAX_BUGS" ,&MAX_BUGS ,300 ,20 ,1000 },
- {PARAMETER ,"INIT_BUGS" ,&INIT_BUGS ,44 ,1 ,100 },
- {PARAMETER ,"INIT_FOOD" ,&INIT_FOOD ,3189 ,1 ,30000 },
- {PARAMETER ,"INIT_ENERGY" ,&INIT_ENERGY ,624 ,10 ,15000 },
- {PARAMETER ,"INIT_VARIANCE",&INIT_VARIANCE ,182 ,1 ,250 },
- // {0 ,"INIT_O2" ,&INIT_O2 ,32000 ,1 ,100000 },
- // {0 ,"INIT_CO2" ,&INIT_CO2 ,4000 ,1 ,100000 },
-
- {PARAMETER ,"MUTATE_RANGE" ,&MUTATE_RANGE ,48 ,1 ,120 },
- {PARAMETER ,"FOOD_RATE" ,&FOOD_RATE ,10 ,1 ,100 },
- {PARAMETER ,"FOOD_ENERGY" ,&FOOD_ENERGY ,624 ,1 ,5000 },
- {PARAMETER ,"X_AND" ,&X_AND ,1 ,1 ,260 },
- {PARAMETER ,"Y_AND" ,&Y_AND ,1 ,1 ,260 },
- {PARAMETER ,"OASIS_SIZE" ,&OASIS_SIZE ,0 ,0 ,50 },
-
- {NULL ,NULL ,NULL ,NULL ,0 ,0 }
- };
-
- // The Command structure array defines commands which AREXX can issue us.
-
- struct command {
- UBYTE flags;
-
- char *name; // name of a REXX command
- RC (*address)(void *); // function's address
- };
-
- RC please_quit (void *); // some forward declarations
- RC restart (void *);
- RC enable_sex (void *);
- RC disable_sex (void *);
- RC enable_vision (void *);
- RC disable_vision (void *);
- RC return_bugs (void *);
- RC return_runs (void *);
- RC return_timesteps(void *);
- RC dumpdefaults (void *);
-
- struct command cmdlist[] = {
- {0 ,"MJ_QUIT" ,&please_quit },
- {0 ,"MJ_RESET" ,&restart },
- {0 ,"MJ_DUMPDEFAULTS" ,&dumpdefaults },
-
- {0 ,"MJ_SEX" ,&enable_sex },
- {0 ,"MJ_NOSEX" ,&disable_sex },
- {0 ,"MJ_VISION" ,&enable_vision },
- {0 ,"MJ_NOVISION" ,&disable_vision },
-
- {0 ,"MJ_BUGS?" ,&return_bugs },
- {0 ,"MJ_RUNS?" ,&return_runs },
- {0 ,"MJ_TIMESTEPS?" ,&return_timesteps },
-
- {0 ,0 ,0 }
- };
-
- //-------------------------------------------------------------------------
-
- struct move_vec {
- short dx; // a vector describing a direction or move.
- short dy;
- };
-
-
- #define NUM_NEIGHBORS 8 // as on a chess board.
-
- // direction vectors for the 8 directions
-
- struct move_vec xymovs[NUM_NEIGHBORS] = {
- {1,0},
- {1,1},
- {0,1},
- {-1,1},
- {-1,0},
- {-1,-1},
- {0,-1},
- {1,-1}
- };
-
- int nbor_offs[NUM_NEIGHBORS] = {
- -1, 1,
- -512, 512,
- -513, 511,
- -511, 513
- };
-
- // exponentially decreasing "scent" weights according to distance
-
- UBYTE scent_vals[]=
- { 254 ,253 ,252 ,251 ,
- 250 ,250 ,250 ,250 ,
- 249 ,249 ,249 ,249 ,
- 248 ,248 ,247 ,247 ,
- 246 ,245 ,244 ,243 ,
- 242 ,240 ,240 ,240 ,
- 235 ,235 ,230 ,230 , // for vision beyond 32 , code will index
- 230 ,210 ,200 ,100}; // beyond this point **!!
-
- char strbuf[80]; // generic string buffer
- char linebuf[512]; // CAPTURE line buffer
- char *lineptr; // ptr into above
-
- FLAG quit_me; // request to terminate program
- FLAG auto_reset; // reset ecosys when bugs=0
- UWORD food_timer; // timer for feeding time.
- UWORD sim_speed; // slow-motion or no delay ?
-
- //-------------------------------------------------------------------------
- #ifdef AMIGA
-
- extern struct ExecBase *SysBase; // basic Kernel services
- extern struct GfxBase *GfxBase; // low-level graphics
- extern struct IntuitionBase *IntuitionBase; // WIMP interface
- struct Library *RexxSysBase; // AREXX support
-
- //-------------------------------------------------------------------------
- // A NewScreen struct to open application Screens.
- //-------------------------------------------------------------------------
-
- struct NewScreen ns = { 0,0, 640, 400, 2,
- 0,1,HIRES|LACE,0,
- NULL,
- "MegaJITTER V1.3 Statistics (c) 1992-1994 L.Vanhelsuwé",
- NULL,NULL};
-
- char scr_name[]="MegaJITTER Ecology Simulator";
-
- struct TextAttr txtattr = {
- "topaz.font", 8, 0,0
- };
-
- struct RastPort *stat_rp;
- struct BitMap *bm; // allocated via AllocBitMap()
-
- struct Screen *eco_screen,*stat_screen;
- struct Window *window;
- struct Menu *menustrip = 0;
- void *vi; // global VisualInfo ptr for Workbench Screen
-
- ULONG *plane0; // ASM routines access this **!!
-
- char version[]="$VER: MegaJitter 1.3 ©LVA 06/MAY/94";
-
- #define REXXPORTNAME "REXX-MJ"
- //-------------------------------------------------------------------------
- // A NewWindow struct to open application Windows.
- //-------------------------------------------------------------------------
-
- struct NewWindow nw = {
- 0, STATS_TOPY,
- STAT_SCR_WIDTH, 200,
- 255, 255, /* Default pens */
-
- // I want to know about following IDCMP Message types
- IDCMP_MENUPICK,
-
- // INVISIBLE window flags
- WFLG_BORDERLESS | WFLG_ACTIVATE,
-
- NULL, // No gadgets in this window
-
- (struct Image *) NULL,
-
- (char *) NULL, // Window title
-
- (struct Screen *) NULL, // to be filled in.
-
- (struct BitMap *) NULL,
-
- 0, 0, /* Minimum sizes */
- 65535, 65535, /* Maximum sizes */
- CUSTOMSCREEN /* and put it IN our screen */
- };
-
- //-------------------------------------------------------------------------
- // Menu Layout in compact GadTools format.
- //
- // **!! NOTE Mutual exclusion masks for Graph speed subitems !
- //-------------------------------------------------------------------------
-
- #define PROJECT_MENU 0
- #define OPTIONS_MENU 1
- #define STATS_MENU 2
- #define HELP_MENU 3
-
- #define PROJ_MENU_RESET 0
- #define PROJ_MENU_LOAD 1
- #define PROJ_MENU_SAVE 2
- #define DUMMY_ITEM0 3
- #define PROJ_MENU_ABOUT 4
- #define PROJ_MENU_AUTHOR 5
- #define PROJ_MENU_QUIT 6
-
- #define OPT_MENU_CTRL_PANEL 0
- #define OPT_MENU_SIM_SPEED 1
- #define OPT_MENU_STAT_SPEED 2
- #define OPT_MENU_RESET 3
- #define OPT_MENU_AUDIT 4
- #define OPT_MENU_MINMAX 5
-
- #define HELP_MENU_a 0
- #define HELP_MENU_b 1
-
- // First empty slot for list of statistics at offset N
-
- #define NEWMENU_AUDIT 22
- #define NEWMENU_APPEND 25
-
- struct NewMenu mymenus[NUM_STATS+NEWMENU_APPEND+2]= { // +END_MARKER + SAFETY
-
- { NM_TITLE, "Project" ,0 ,0 ,0 ,0},
- { NM_ITEM, "Reset" ,"N" ,0 ,0 ,0}, // Big Bang !
- { NM_ITEM, "Load..." ,"L" ,0 ,0 ,0}, // Load
- { NM_ITEM, "Save..." ,"S" ,0 ,0 ,0}, // Save
- { NM_ITEM, NM_BARLABEL ,0 ,0 ,0 ,0},
- { NM_ITEM, "About.." ,"?" ,0 ,0 ,0},
- { NM_ITEM, "Author..." ,0 ,0 ,0 ,0},
- { NM_ITEM, "Quit" ,"Q" ,0 ,0 ,0},
-
- { NM_TITLE, "Control" ,0 ,0 ,0 ,0},
- { NM_ITEM, "Global Parameters..." ,0 ,0 ,0 ,0},
- { NM_ITEM, "Simulation Speed" ,0 ,0 ,0 ,0},
- { NM_SUB, "Full Speed" ,"F",CHECKED|CHECKIT|MENUTOGGLE ,0x02 ,0},
- { NM_SUB, "Slow-Motion" ,"Z", CHECKIT|MENUTOGGLE ,0x01 ,0},
- { NM_ITEM, "Statistics Speed" ,0 ,0 ,0 ,0},
- { NM_SUB, "Ultra Slow" ,"1", CHECKIT|MENUTOGGLE ,0x7E ,0},
- { NM_SUB, "Very Slow" ,"2", CHECKIT|MENUTOGGLE ,0x7D ,0},
- { NM_SUB, "Slow" ,"3", CHECKIT|MENUTOGGLE ,0x7B ,0},
- { NM_SUB, "Normal" ,"4",CHECKED|CHECKIT|MENUTOGGLE ,0x77 ,0},
- { NM_SUB, "Fast" ,"5", CHECKIT|MENUTOGGLE ,0x6F ,0},
- { NM_SUB, "Very Fast" ,"6", CHECKIT|MENUTOGGLE ,0x5F ,0},
- { NM_SUB, "Ultra Fast" ,"7", CHECKIT|MENUTOGGLE ,0x3F ,0},
- { NM_ITEM, "Reset on Extinction" ,"R",CHECKED|CHECKIT|MENUTOGGLE ,0 ,0},
- { NM_ITEM, "Toggle Audit Trail" ,"A", CHECKIT|MENUTOGGLE ,0 ,0},
- { NM_ITEM, "Toggle Minima/Maxima","M", CHECKIT|MENUTOGGLE ,0 ,0},
-
- { NM_TITLE, "Statistics" ,0 ,0 ,0 ,0}
- // Here we dynamically append the statistics menu items...
- };
-
- BPTR capture_file; // **!! not used
-
- struct MsgPort *rexx_port;
-
- #endif // AMIGA OS specific structs
-
- //-------------------------------------------------------------------------
- // ---------------------------
- // Now the function prototypes
- // ---------------------------
-
- void handle_bug (BUGPTR creat);
- void spawn_child (BUGPTR mother, BUGPTR father, FLAG sexy);
- void grow_food (void);
-
- void open_libraries (void);
- void init_defaults (struct variable *varptr);
- void init_gfx (void);
- void alloc_structs (void);
-
- void close_ecosys (void);
- void close_gfx (void);
- void wipe_environment (void);
- FLAG reset_ecosystem (void);
-
- void reset_global_stats (void);
- void collect_bug_stats (BUGPTR creat);
- void update_statistics (void);
- void print_stat (struct statistic *stat, ULONG num, int x);
- void print_minmax (struct statistic *stat, UBYTE value, int x);
-
- void append_stats_menus (void);
- void toggle_stat (void);
- void handle_IDCMP_msgs (void);
- void handle_AREXX_msgs (void);
- void handle_menuselection (SHORT menu);
- void uncheck_menuitem (int menunum, int itemnum);
- void popup_About_req (void);
- void popup_Author_req (void);
- void PaintStatLabels (void);
- void PaintStatValues (void);
- void RePaintGraphs (void);
- void WriteString (UWORD x, UWORD y, char *string);
- void close_REXX (void);
-
- FLAG kill_switch (void);
- FLAG get_user_options (void);
- FLAG menuitem_checkstate (int menunum, int itemnum);
- FLAG post_REXX_port (char * portname);
- RC parse_rexx_cmd (struct RexxMsg* msg);
-
- UBYTE mutate (UBYTE gene);
- UBYTE mix (UBYTE gene1, UBYTE gene2);
-
- int quick_req (char *message, char *exit);
- short rnd (void);
-
- // The program has two fundamentally different ways of doing things:
- // 1) It uses colored pixels in a visible screen to store the ecosystem.
- // 2) It uses an invisible BYTE-map to hold the ecosystem.
- //
- // The program itself is unaware of this difference by using function pointers
- // to 2 sets of routines doing the same thing, but on 2 different representations.
-
- void (*cell_writer) (ULONG x, ULONG y, ULONG cell, BUGPTR bug);
- void (*cell_eraser) (BUGPTR bug);
- CELL (*cell_typer) (ULONG x, ULONG y);
- FLAG (*food_checker) (ULONG x, ULONG y);
- FLAG (*clear_checker) (ULONG x, ULONG y);
-
- #define set_cell_to(x,y,cell,bug) (*cell_writer) ((x),(y),(cell),(bug))
- #define erase_cell(bug) (*cell_eraser) (bug)
- #define cell_type(x,y) (*cell_typer) ((x),(y))
- #define is_food(x,y) (*food_checker) ((x),(y))
- #define is_empty(x,y) (*clear_checker)((x),(y))
-
- void blind_put_cell (ULONG x, ULONG y, ULONG cell, BUGPTR bug);
- void gfx_put_cell (ULONG x, ULONG y, ULONG cell, BUGPTR bug);
- extern void asm_plot_pixel (struct fast_pixel *fp, ULONG x, ULONG y, UBYTE color);
-
- void blind_wipe_cell (BUGPTR bug);
- void gfx_wipe_cell (BUGPTR bug);
- extern void asm_fastwipe_pixel (struct fast_pixel *fp);
-
- CELL blind_cell_type (ULONG x, ULONG y);
- extern CELL asm_read_pixel (ULONG x, ULONG y);
-
- FLAG blind_is_food (ULONG x, ULONG y);
- extern FLAG asm_is_food (ULONG x, ULONG y);
-
- FLAG blind_is_empty (ULONG x, ULONG y);
- extern FLAG asm_is_empty (ULONG x, ULONG y);
-
- //============================================================================
- //
- // MAIN()
- //
- // The main loop consists of:
- //
- // - resetting per-loop statistics
- // - letting all live creatures do their thing once (& accumulating stats)
- // - now and again dropping some food.
- // - now and again updating the scrolling graphs with simualtion statistics
- //
- //============================================================================
-
- void main(int argc, char **argv) {
-
- UWORD i;
- BUGPTR creat; // pointer to current creature
-
- if (sizeof(struct bug) & 3)
- printf("Warning : Bug Structure is not LONG-word-multiple sized (sizeof: %d)\n", sizeof(struct bug));
-
- // This program needs at least OS version 2.0 and a Motorola 68020 CPU.
-
- if ( ((struct Library*)SysBase)->lib_Version < 39) {
- printf("I'm very sorry, but this program needs at least OS V39.\n");
- exit(FATAL_ERROR);
- }
-
- if ( (SysBase->AttnFlags & AFF_68020) == 0 ) {
- printf("I'm very sorry, but this program needs at least a 68020 CPU.\n");
- exit(FATAL_ERROR);
- }
-
- printf("MegaJitter V1.3 (C) 1992-94 Laurence Vanhelsuwé.\n");
- printf("An Evolution Simulator.\n\n");
-
- if (FindPort(REXXPORTNAME)) {
- printf("MegaJitter is already running!\n\n");
- printf("You can control the running program by executing AREXX scripts which\n");
- printf("communicate with MJ through the \"ADDRESS '" REXXPORTNAME "'\" command.\n");
- exit(FATAL_ERROR);
- }
-
- // Grab Intuition, Graphics, REXX, ... function libraries.
-
- open_libraries();
-
- if (get_user_options()) { // ARGUMENT PARSING done Amiga-specific !
-
- printf("Error in user selected options. Please consult documentation.\n");
-
- close_gfx();
- exit(FATAL_ERROR);
- }
-
- if (!post_REXX_port(REXXPORTNAME)) { // create public REXX (input) port
- printf("Couldn't open AREXX communications port!\n");
-
- close_gfx();
- exit(FATAL_ERROR);
- }
-
- init_gfx(); // init graphics-related stuff.
-
- alloc_structs(); // allocate data structs
-
- reset_ecosystem(); // init eco data structs
-
- SetTaskPri(FindTask(0L), -5); // lower our priority: be user friendly
-
- births = deaths = 0; // update_stats resets
-
- while ( !kill_switch() ) { // JOYSTICK FIRE BUTTON ends !!
-
- if (sim_speed) Delay(sim_speed); // slow-motion or not ?
-
- IFAUDIT printf("T = %5d\n", time);
-
- time++; // time flows in Universe..
-
- reset_global_stats(); // reset some statistics counters
-
- // modulate daylight using time counter.
- daylight = 50+ (WORD) (50 * cos( ((double)time) / 4000 ) );
-
- //----------------------------------------------------------------
- // Process all creatures... let them live.
- //----------------------------------------------------------------
-
- creat = &bugs[0]; // -> 1st bug in ecosystem
- for (i=0; i< MAX_BUGS; i++, creat++) { // scan array for living bugs
- if (creat->alive) {
- handle_bug(creat); // let creature do its things
- }
- }
-
- //----------------------------------------------------------------
- // Now collect some statistics from all live bugs..
- //----------------------------------------------------------------
-
- numbugs = 0;
- creat = &bugs[0]; // -> 1st bug in ecosystem
- for (i=0; i< MAX_BUGS; i++, creat++) { // scan array for living bugs
- if (creat->alive) {
- numbugs++;
- collect_bug_stats(creat); // then collect some figures..
- }
- }
- //----------------------------------------------------------------
- // If everything died and auto_reset feature is ON: restart all.
- //----------------------------------------------------------------
-
- if (!numbugs && auto_reset) reset_ecosystem();
-
- //----------------------------------------------------------------
- // Generate food for creatures.
- //----------------------------------------------------------------
-
- if (food_timer--) {}
- else {
- food_timer = FOOD_RATE; // reset counter.
- grow_food(); // and drop a parcel of manna.
- grow_food(); // twice
- }
-
- update_statistics(); // update stats graphs
-
- handle_IDCMP_msgs(); // check our external events...
-
- if (RexxSysBase) handle_AREXX_msgs(); // check rexx Msgs only if REXX active
-
- } // END OF MAIN LOOP
-
-
- // God pressed fire button on joystick: kill off Universe !
-
- close_ecosys(); // dealloc data structs
- close_gfx(); // close screens.
- close_REXX();
- SetTaskPri(FindTask(0L), 0); // restore standard CLI priority
-
- exit(0); // don't set any return code
- }
- //===============================================================
- // CREATURE ROUTINE
- // ----------------
- // This routine is called for every live creature.
- // It does some things unconditionally to all creatures like
- // - ageing
- // - exhausting
- //
- // and does other things in a gene-dependent way like
- // - moving
- // - seeing
- // - reproducing
- // - dying of old age
- //
- //===============================================================
-
- void handle_bug (BUGPTR creat) {
-
- UBYTE ran, rot, color;
- register int offs, x,y, dx,dy, coordmask;
- int vision,scent,distance, hurdle;
- short i;
- CELL cell;
- BUGPTR *sexy,*neighb, partner; // ptr to bugs wanting sex.
- BUGPTR victim;
-
- creat->age++; // creatures age in sync with Time...
- creat->energy--; // energy is constantly going down
-
- #ifndef FASTER
- IFAUDIT if (creat->energy < 100)
- printf("Bug #%3d is dying of exhaustion (E: %d)\n", creat-bugs, creat->energy);
- #endif
-
- // If creature exhausts energy or its designed lifespan... it dies.
-
- if (creat->energy <= 0) {
-
- creat->alive = FALSE;
- erase_cell(creat); // then creature dies...
- deaths++; // track morbidity rate
- #ifndef FASTER
- IFAUDIT printf("Bug #%3d died of lack of energy.\n", creat-bugs);
- #endif
-
- return;
-
- } else
-
- // If creature beyond optimum life span, start throwing dice but bias survival
- // with age overrun and creatures' energy (the fittest have a better chance).
-
- if (creat->age >= (creat->dying_age <<8 )) {
-
- #ifndef FASTER
- IFAUDIT printf("Bug #%3d is dying of old age (AGE: %d > %d)", creat-bugs, creat->age, creat->dying_age<<8);
- #endif
-
- hurdle = creat->age - (creat->dying_age <<8); // negative bias
- hurdle -= creat->energy; // positive bias
- if (hurdle < 0) hurdle = 0;
-
- if (( rnd() & 0x1FFF) < hurdle) { // dice value upto 8191
-
- creat->alive = FALSE;
- #ifndef FASTER
- IFAUDIT printf("Died of old age (ENERGY: %d).\n", creat->energy);
- #endif
- erase_cell(creat); // then creature dies...
- deaths++; // track morbidity rate
- return;
-
- }
- #ifndef FASTER
- else
- IFAUDIT printf("But is hanging on... (ENERGY: %d).\n", creat->energy);
- #endif
- }
-
- // Control speed of each creature.
-
- if (creat->speed_delay >= 1) { // while speed delay counter ticks...
- creat->speed_delay--; // don't move.
- return;
- }
- creat->speed_delay = creat->speed >> 5; // 0..7
-
- // At this point, our creature decides it's gonna move.
- // This entails an energy penalty.
-
- creat->energy -= 2; // moving is N times harder than not moving
-
- // Check energy exhaustion again.
-
- if (creat->energy <= 0) {
- creat->alive = FALSE;
- #ifndef FASTER
- IFAUDIT printf("Bug #%3d died of lack of energy when moving.\n", creat-bugs);
- #endif
-
- erase_cell(creat); // then creature dies...
- deaths++; // track morbidity rate
- return;
- }
-
- // Here's the engine of biological Evolution: when a creature reaches a certain
- // level of fitness, it wants to procreate.
- // Procreation occurs in two modes:
- //
- // 1) ASEXUAL: it just "splits". The child creature is slightly mutated.
- // 2) SEXUAL: the creature looks for a neighbor who's also "sex-ripe"
- //
- // This is the mechanism that allows stronger (and weaker) forms to emerge.
- // The environment will determine the statistical chances of survival and thus
- // the path of evolution and extinctions.
-
- if (creat->energy > creat->procr_energy<<6 ) {
-
- #ifndef FASTER
- IFAUDIT printf("Bug #%3d Needs to procreate (E=%4d > %4d)\t", creat-bugs, creat->energy, creat->procr_energy<<6);
- #endif
-
- if (creat->age < (creat->minsexage<<4)) {
- #ifndef FASTER
- IFAUDIT printf("But is too young...(AGE: %d < %d)\n", creat->age, creat->minsexage<<4);
- #endif
- } else
-
- if (creat->age > (creat->maxsexage<<8)) {
- #ifndef FASTER
- IFAUDIT printf("But is too old...(AGE: %d > %d)\n", creat->age, creat->maxsexage<<8);
- #endif
- } else
-
- if (!SEX) {
- #ifndef FASTER
- IFAUDIT printf("Just splitting...\n");
- #endif
- spawn_child(creat, (BUGPTR) 0, ASEXUAL);
- } else
-
- if ( (UBYTE)rnd() < creat->sexuality ) {
- #ifndef FASTER
- IFAUDIT printf("Looking for mate...");
- #endif
- sexy = owner_ptrs + creat->x + (creat->y <<9);
-
- // Check all around randy bug to see if there's another bug with
- // which it can mate.
-
- for(i=0; i<NUM_NEIGHBORS; i++) {
-
- offs = nbor_offs[i];
-
- // Make sure we don't index into out-of-bounds RAM
- // This could occur at the very top or the very bottom of
- // the array, optionally we could AND **!!
-
- if (offs > 0 ) {
- if (creat->y == ACRE) continue;
- } else {
- if (creat->y == 0) continue;
- }
-
- neighb = sexy + offs;
-
- if (partner = *neighb) {
-
- if (partner < bugs || partner > &bugs[MAX_BUGS]) {
- printf("FATAL BUG: owner_ptr (%08X) doesn't point back to a creature !!\n", partner);
- printf("Mother #%d (x,y)= (%d,%d)\n", creat-bugs, creat->x, creat->y);
- printf("Trying to mate with offset %d (%d)\n", i, nbor_offs[i]);
- exit(PROG_BUG_ERROR);
- }
-
- #ifndef FASTER
- IFAUDIT printf("\nPartner found @ (%d,%d)\n", partner->x, partner->y);
- #endif
-
- // Select partners which are strong and healthy !!
-
- if (partner->energy > creat->energy) {
-
- #ifndef FASTER
- IFAUDIT {
- printf("Sex ! (BUGS: %d & %d, ENERGIES: %d & %d)\n", creat-bugs, partner-bugs, creat->energy, partner->energy);
-
- printf("Mother (x,y)= (%d,%d)\n", creat->x, creat->y);
- printf("Father (x,y)= (%d,%d)\n", partner->x, partner->y);
- }
- #endif
- spawn_child(creat, partner, SEXUAL);
- break;
-
- } else IFAUDIT printf("Partner to weak ! (%d < %d)\n", partner->energy, creat->energy);
- }
- }
-
- IFAUDIT if (i==NUM_NEIGHBORS) printf("Failed to locate.\n");
-
- } else {
- #ifndef FASTER
- IFAUDIT printf("Just splitting...\n");
- #endif
- spawn_child(creat, (BUGPTR) 0, ASEXUAL);
- }
- } // IF ENERGY > PROCR_ENERGY
-
-
- // This section implements food vision (if not disabled)
- // A creature's vision gets modulated by the time of day.
- // 0.. 25 = total darkness (blind)
- // 25..100 = 0..100% vision.
-
- scent = 0; // accumulated scent = 0
-
- if (VISION) {
- if (daylight > 25) { // if it's day ... (night = total dark)
-
- vision = creat->vision>>2; // shrink 0..255 gene to 0..63 length range
-
- i = vision = (int) ((float) vision * (float) (daylight-25) / 75.0);
-
- distance = 0; // set distance index to 0
-
- dx = xymovs[creat->dir].dx; // look in direction of
- dy = xymovs[creat->dir].dy; // current movement.
-
- coordmask = ACRE; // cache andmask 511 in register
-
- x = (creat->x +dx) & ACRE;
- y = (creat->y +dy) & ACRE;
-
- while(i--) {
- if ( ! is_empty(x, y))
- scent += scent_vals[distance];
-
- x = (x+dx) & coordmask; // & ACRE to ensure we don't look
- y = (y+dy) & coordmask; // beyond world boundary **!!
- distance++;
- }
-
- #ifndef FASTER
- IFAUDIT if (scent) printf("Bug #%3d detected food with a vision of %3d (scent: %d)\n", creat-bugs, vision, scent);
- #endif
- }
- }
-
- // This section of code determines how creatures move.
- // The strength of the scent LOWERS the probability of doing a gene-induced turn
-
- rot = 0; // clear rotation accumulator
-
- ran = (UBYTE)rnd(); // throw a dice
- if (ran > scent) { // if number over scent hurdle...
- if ((UBYTE)rnd() > creat->right ) rot = rot + 1;
- if ((UBYTE)rnd() > creat->left ) rot = rot - 1;
-
- creat->dir += rot;
- creat->dir &= 7; // force into 0..7
-
- #ifndef FASTER
- IFAUDIT if (VISION && scent) printf(" .. And deviated from scent path !! (%d > %d)\n", ran, scent);
- #endif
- }
-
- // Calculate tentative forward step (x,y)
- x = (creat->x + xymovs[creat->dir].dx) & ACRE; // move into 1 of 8 dirs
- y = (creat->y + xymovs[creat->dir].dy) & ACRE;
-
- // If the creature is about to move onto a pixel with food on, "eat it" !
- // Food can be either another bug or simple vegetation.
- // If Bug is mainly a Veggie, there's a strong bias to refusing meat.
- // If Bug is mainly a meat eater, there's a strong bias to refusing vegetation.
-
- cell = cell_type(x, y);
- //printf("Cell type bug #%3d is moving towards: %d @ (%d,%d)\n", creat-bugs, cell, x,y);
- switch (cell) {
-
- case VEGE_CELL:
- if ((UBYTE)rnd() > creat->foodtype) { // small FOODTYPE = VEGETARIAN
-
- IFAUDIT printf("Bug #%3d Eats Vegetation @ (%d,%d) (+%d E, NEW: %d) (GENE=%3d% (%s))\n",
- creat-bugs,x,y, FOOD_ENERGY, creat->energy + FOOD_ENERGY,
- creat->foodtype*100/256,
- creat->foodtype>128?"CARNIVORE":"HERBIVORE");
-
- creat->energy += FOOD_ENERGY; // if food found, eat it.
- food_in_system--; // track amount of food left
- } else {
- IFAUDIT printf("Bug #%3d avoids Vegetation (GENE=%3d% (%s))\n",
- creat-bugs, creat->foodtype*100/256,
- creat->foodtype>128?"CARNIVORE":"HERBIVORE");
-
- break; // don't fall through. Food obstructs move !
- }
-
- // Now fall through as if moving onto empty cell. **!!
-
-
- case NEUT_CELL:
- erase_cell(creat); // erase old position
-
- creat->x = x; // move to new cell position
- creat->y = y;
-
- // color = 1 + (creat->energy >> color_divider);
- // color = color < VEGE_CELL ? color : VEGE_CELL-1;
- color = VEGE_CELL-1;
-
- set_cell_to(x,y, color, creat);
- if ( *(owner_ptrs+x+(y<<9)) ) {
- BUGPTR rogue;
-
- rogue = *(owner_ptrs+x+(y<<9));
-
- printf("FATAL BUG: supposedly EMPTY cell contains an owner back-ptr !\n");
- printf("Dangling back-ptr : %08X (bug # %d)\n", rogue, rogue-bugs);
- printf("Corresponds to following bug:\n");
- printf("%s, (%d,%d), (empty cell x,y = %d,%d)\n",
- rogue->alive?"ALIVE":"DEAD", rogue->x, rogue->y, x, y);
- }
-
- *(owner_ptrs+x+(y<<9)) = creat; // update parallel array
- break;
-
- case WALL_CELL: // don't move in case of wall !
- break;
-
- default: // default case SHOULD mean bumping into bug...
-
- victim = *(owner_ptrs + x + (y<<9));
-
- if (victim) {
- if ((UBYTE)rnd() < creat->foodtype) { // large FOODTYPE = CARNIVORE
-
- IFAUDIT printf("Bug #%3d wants to eat bug #%3d @ (%d,%d) (GENE=%3d% (%s))\n",
- creat-bugs,victim-bugs,x,y,
- creat->foodtype*100/256,
- creat->foodtype>128?"CARNIVORE":"HERBIVORE");
-
- if (victim->energy < creat->energy) {
-
- creat->energy += victim->energy; // eat weaker creature
-
- IFAUDIT printf("Bug #%3d Eats Bug #%3d (+%d E NEW: %d)!\n",
- creat-bugs, victim-bugs, victim->energy, creat->energy);
-
- victim->alive = FALSE; // kill victim
- deaths++;
- erase_cell(victim); // erase old position
-
- erase_cell(creat); // erase old position
- creat->x = x; // move to new cell position
- creat->y = y;
-
- // color = 1 + (creat->energy >> color_divider);
- // color = color < VEGE_CELL ? color : VEGE_CELL-1;
- color = VEGE_CELL-1;
-
- set_cell_to(x,y, color, creat);
-
- *(owner_ptrs+x+(y<<9)) = creat; // update parallel array
-
- } else {
- IFAUDIT printf("Bug #%3d Avoids Eating Bug #%3d because 2nd bug is stronger!\n",
- creat-bugs, victim-bugs);
- }
-
- } else {
- IFAUDIT printf("Bug #%3d avoids Bug @ (%d,%d) (+%d E, NEW: %d) (GENE=%3d% (%s))\n",
- creat-bugs,x,y, FOOD_ENERGY, creat->energy + FOOD_ENERGY,
- creat->foodtype*100/256,
- creat->foodtype>128?"CARNIVORE":"HERBIVORE");
- }
- }
- }
- }
- //----------------------------------------------------------------
- // OLD Oxygen stuff originally straight after age++;
- //----------------------------------------------------------------
- // oxygen += creat->oxy_produce; // discharge by-product O2
-
- // if (oxygen >= creat->oxy_consume) { // if O2-breathing, and if there's
- // oxygen -= creat->oxy_consume; // enough O2 : grab your quota
- // } else creat->energy = 0; // otherwise you suffocate
- //
- // carb_dioxide += creat->co2_produce;
- // if (carb_dioxide >= creat->co2_consume) { // idem with CO2
- // carb_dioxide -= creat->co2_consume;
- // } else creat->energy = 0;
-
-
-
- //---------------------------------------------------------------------------
- // This routine handles the details of giving birth to a new creature.
- //
- // 1) ASEXUAL
- // Basically, a dead bug structure is found and then the parent is cloned
- // in this space. The gene variables of the parent are slightly mutated
- // in the child.
- //
- // 2) SEXUAL
- // A dead bug structure is found and then the child genes are created from
- // a mixture of both parent's genes.
- //
- // The energy exchange which happens is as follows:
- // - the child gets 30 % of the mother's energy
- // - the mother's energy goes down to 70%
- //
- // For sexual reproduction both parents drop to 85% of their previous level.
- //---------------------------------------------------------------------------
-
- void spawn_child (BUGPTR mother, BUGPTR father, FLAG sexy) {
-
- BUGPTR child;
-
- int i;
-
- child = &bugs[0];
-
- for (i=0; i< MAX_BUGS; i++, child++) { // scan array to find empty slot..
-
- if (!child->alive) {
-
- births++; // record birth
-
- *child = *mother; // copy mother's genes for starters..
- // This means:
- // child->x == mother->x
- // child->y == mother->y
- // child->prevpix == mother->prevpix
-
-
- child->age = 0; // new-born !
- child->generation++; // a new generation !
-
- if (sexy) {
- child->energy = (WORD) (
- 30 * ( (int) mother->energy +
- (int) father->energy ) / 100) ;
- mother->energy = (WORD) ((85 * (int)mother->energy) / 100);
- father->energy = (WORD) ((85 * (int)father->energy) / 100);
-
- child->right = mix(mother->right, father->right);
- child->left = mix(mother->left, father->left);
- child->vision = mix(mother->vision, father->vision);
- child->sexuality = mix(mother->sexuality,father->sexuality);
- child->dirgene = mix(mother->dirgene, father->dirgene);
- child->dying_age = mix(mother->dying_age,father->dying_age);
- child->minsexage = mix(mother->minsexage,father->minsexage);
- child->maxsexage = mix(mother->maxsexage,father->maxsexage);
- child->speed = mix(mother->speed, father->speed);
- child->foodtype = mix(mother->foodtype, father->foodtype);
- child->procr_energy = mix(mother->procr_energy, father->procr_energy);
- } else {
- child->energy = (WORD) ((30 * (int)mother->energy) / 100);
- mother->energy = (WORD) ((70 * (int)mother->energy) / 100);
-
- child->right = mutate (child->right);
- child->left = mutate (child->left );
- child->vision = mutate (child->vision);
- child->sexuality = mutate (child->sexuality);
- child->dirgene = mutate (child->dirgene);
- child->dying_age = mutate (child->dying_age);
- child->minsexage = mutate (child->minsexage);
- child->maxsexage = mutate (child->maxsexage);
- child->speed = mutate (child->speed);
- child->foodtype = mutate (child->foodtype);
- child->procr_energy = mutate (child->procr_energy);
- }
-
- // whizz away from mother in a gene dependent way
- child->dir = (child->dir +(child->dirgene>>5)) % 8;
-
- child->speed_delay = child->speed >> 6; // 0..3
-
- return;
- }
- }
- }
- //----------------------------------------------------------------
- // Slightly modify a gene (a variable in the unsigned range 0..255).
- // The variable is not allowed to wrap-around, but instead "hits the
- // ceiling".
- //
- // Note that the mutation coding is subtle. The mutation should not
- // have any bias towards lowering or increasing the gene's value **!!
- // It should be STRICTLY random !
- //----------------------------------------------------------------
- UBYTE mutate (UBYTE gene) {
-
- int newgene;
-
- newgene = gene + rnd()%MUTATE_RANGE - rnd()%MUTATE_RANGE;
-
- if (newgene > 255) return 255;
- else
- if (newgene < 0 ) return 0;
-
- return (UBYTE) newgene;
- }
- //----------------------------------------------------------------
- // Mix the 2 genes of two parent creatures to produce a child's gene.
- //----------------------------------------------------------------
- UBYTE mix (UBYTE gene1, UBYTE gene2) {
-
- return mutate( (UBYTE) (( (UWORD)gene1 + (UWORD)gene2 ) /2 ));
- }
-
- //----------------------------------------------------------------
- // Drop some food in our world on a spot which IS NOT YET FOOD.
- // We can't put food on food OR food on a bug otherwise our food stats
- // will be inaccurate.
- //----------------------------------------------------------------
- void grow_food(void) {
-
- short fx,fy;
- UBYTE food=VEGE_CELL;
- int dummy;
- register int attempts;
- double ra,rr; // random angle (in rads)
-
- // modulate food growth with night/day
- // From 0..50 modulate growth between 50% and 100%
- // Between 51..100 food growth is always 100%
-
- if (rnd()%100 > (daylight+50)) return;
-
- attempts = 20; // attemps at finding empty spot..
-
- while (attempts--) {
- if (OASIS_SIZE) {
-
- ra = (double) rnd(); // pick a random dot inside a circle
- rr = (double) (rnd() % OASIS_SIZE);
-
- fx = ACRE/2 + (int) (cos(ra) * rr);
- fy = ACRE/2 + (int) (sin(ra) * rr);
-
- } else {
- fx = (rnd()%ACRE) & (-X_AND);
-
- dummy = fx * fx * attempts; // introduce random delay
- dummy /= 20; // randomness stems from data-depen
-
- fy = (rnd()%ACRE) & (-Y_AND);
- }
-
- fx &= ACRE; // ensure we stay within the ACRE
- fy &= ACRE;
-
- if (is_empty(fx, fy)) {
- set_cell_to(fx,fy, food, &dummy_bug);
- food_in_system++; // track amount of food in our world
- break;
- }
- }
- }
- //----------------------------------------------------------------
- // Gain access to system function libraries (Dynamic Link Libraries).
- //----------------------------------------------------------------
- void open_libraries(void) {
-
- IntuitionBase = (void*) OpenLibrary("intuition.library",0);
- if (!IntuitionBase) {
- printf("MAJOR PROBLEM: Couldn't open INTUITION library!\n");
- exit(FATAL_ERROR);
- }
-
- GfxBase = (void*) OpenLibrary("graphics.library",0);
- if (!GfxBase) {
-
- CloseLibrary((struct Library *) IntuitionBase);
-
- printf("MAJOR PROBLEM: Couldn't open GRAPHICS library!\n");
- exit(FATAL_ERROR);
- }
-
- RexxSysBase = (void*) OpenLibrary("rexxsyslib.library",0);
- if (!RexxSysBase) {
-
- printf("AREXX Not available. All of MegaJitter's AREXX functions are disabled.\n");
- printf("If you do have AREXX on your system, please run RexxMast to start AREXX,\n");
- printf("next time before you start MegaJitter.\n\n");
- }
- }
- //----------------------------------------------------------------
- // Check any user options.
- //
- // For example, the program can be evoked as follows:
- //
- // 1> MJ INIT_FOOD 2000 INIT_BUGS 20 SEX AUDIT
- //----------------------------------------------------------------
-
- FLAG get_user_options( void ) {
-
- struct RDArgs *rdargs;
- struct variable *var;
-
- #ifndef DEBUG
- if (! (rdargs = ReadArgs(MJIT_DOS_TEMPLATE, (ULONG*) &options, NULL))) {
- printf("ERROR: ARGUMENTS INCORRECT.\n");
- printf("Type MegaJitter ? for a full syntax template.\n\n");
-
- return SYNTAX_ERROR;
-
- } // Free RDA only AFTER we've parsed ptrs in options pointing into RDA **!!
-
- #endif
-
- // Initialize device independent cell handlers (visual routines are the default)
-
- if (NOANIM) {
- cell_writer = blind_put_cell; // all of these are POINTERS to
- cell_eraser = blind_wipe_cell; // routines.
- cell_typer = blind_cell_type;
- food_checker = blind_is_food;
- clear_checker = blind_is_empty;
- } else {
- cell_writer = gfx_put_cell;
- cell_eraser = gfx_wipe_cell;
- cell_typer = asm_read_pixel;
- food_checker = asm_is_food;
- clear_checker = asm_is_empty;
- }
-
- // If user didn't select the NTSC option and MJ thinks it is running on an
- // NTSC machine, ask user if it wishes to switch NTSC compatibility mode on.
-
- if (!NTSC && SysBase->ex_EClockFrequency != 709379 ) {
- if (quick_req("I think this is an NTSC machine!", "CORRECT|NOPE"))
- NTSC = TRUE;
- }
-
- // Any non-initialized simulation globals get initialized now.
-
- init_defaults(&varlist[0]); // others are intialized from CMD line.
-
- // **!! NOW we can free RDA (AFTER we've parsed ptrs)
-
- FreeArgs(rdargs);
-
- // Most sanity checks can be performed from pre-set data...
-
- var = &varlist[0];
- while(var->name) {
-
- if (*var->address > var->max_val) {
- printf("%s has to be in the range [%d..%d] ! (%d > %d)\n",
- var->name, var->min_val, var->max_val, *var->address, var->max_val);
- exit(SYNTAX_ERROR);
- } else
-
- if (*var->address < var->min_val) {
- printf("%s has to be in the range [%d..%d] ! (%d < %d)\n",
- var->name, var->min_val, var->max_val, *var->address, var->min_val);
- exit(SYNTAX_ERROR);
- }
-
- var++; // goto next var in list
- }
-
- // Check user-initialized parameters for outrageous values.
- // Inform user of range of variables if out of bounds.
-
- if (INIT_BUGS > MAX_BUGS) {
- printf("INIT_BUGS can not be larger than MAX_BUGS! (%d > %d)\n",
- INIT_BUGS, MAX_BUGS);
- return SYNTAX_ERROR;
- }
-
- // Copy MAX_BUGS into full scale value field of numbugs statistic.
-
- available_stats[0].scale = MAX_BUGS;
-
- // Show user the resulting scenario settings, if requested.
-
- if (SHOWDEFAULTS) {
-
- dumpdefaults((void*)0);
-
- printf("\n\nHit ENTER to continue...\n");
- getch();
- }
-
- // If we want to capture all simulation data & statistics, open recording file.
-
- if (CAPTURE) {
- capture_file = Output(); // **!! TEMPORARY
- }
-
- return 0;
- }
-
- //----------------------------------------------------------------
- //
- //----------------------------------------------------------------
- RC dumpdefaults(void* dummy) {
-
- printf("audit = %d (%08X)\n",REPORT,REPORT);
- printf("capture = %d (%08X)\n",CAPTURE,CAPTURE);
- printf("noanim = %d (%08X)\n",NOANIM,NOANIM);
- printf("vision = %d (%08X)\n",VISION,VISION);
- printf("sex = %d (%08X)\n",SEX,SEX);
- printf("MAX_BUGS = %d (%08X)\n",MAX_BUGS,MAX_BUGS);
- printf("\n");
- printf("INIT_BUGS = %d (%08X)\n",INIT_BUGS,INIT_BUGS);
- printf("INIT_FOOD = %d (%08X)\n",INIT_FOOD,INIT_FOOD);
- printf("INIT_ENERGY = %d (%08X)\n",INIT_ENERGY,INIT_ENERGY);
- printf("INIT_VARIANCE = %d (%08X)\n",INIT_VARIANCE,INIT_VARIANCE);
- printf("MUTATE_RANGE = %d (%08X)\n",MUTATE_RANGE,MUTATE_RANGE);
- printf("FOOD_RATE = %d (%08X)\n",FOOD_RATE,FOOD_RATE);
- printf("FOOD_ENERGY = %d (%08X)\n",FOOD_ENERGY,FOOD_ENERGY);
- printf("X_AND = %d (%08X)\n",X_AND,X_AND);
- printf("Y_AND = %d (%08X)\n",Y_AND,Y_AND);
- printf("OASIS_SIZE = %d (%08X)\n",OASIS_SIZE,OASIS_SIZE);
- // printf("INIT_O2 = %d (%08X)\n",INIT_O2,INIT_O2);
- // printf("INIT_CO2 = %d (%08X)\n",INIT_CO2,INIT_CO2);
-
- return TRUE;
- }
-
- //----------------------------------------------------------------
- // Set all non-initialized Simulation Global Parameters to a default value
- // IF the user hasn't given a value himself.
- //----------------------------------------------------------------
- void init_defaults (struct variable *varptr) {
-
- ULONG *var;
-
- while (varptr->name) {
- var = varptr->address;
-
- if (*var) { // if RDA initialized a ptr to an arg,
- *var = atoi((char*) *var); // convert ASCII arg to number
- } else {
- *var = varptr->init_val; // else use built-in default value.
- }
-
- varptr++; // goto next variable
- }
- }
- //----------------------------------------------------------------
- // Before every main loop : reset and recalc some global vars.
- //----------------------------------------------------------------
- void reset_global_stats() {
-
- //int i;
-
- // Calculate the mapping factor to go from
- // creat->energy to creature pixel/cell number (as a right shift)
-
- // color_divider = (MAX_ENERGY+FOOD_ENERGY) / MAX_COLORS;
-
- // for (i=0; i< 30; i++) { // determine howmuch to right shift
- // if (color_divider) {
- // color_divider >>= 1;
- // } else break;
- // }
-
- // i--;
- // color_divider = i;
- // printf("SHIFTS = %d\t\t\t%d>>%d=%d\n\n", i, MAX_ENERGY+FOOD_ENERGY, i, (MAX_ENERGY+FOOD_ENERGY)>