home *** CD-ROM | disk | FTP | other *** search
/ Chestnut's Multimedia Mania / MM_MANIA.ISO / graphics / povsrc20 / parse.c < prev    next >
C/C++ Source or Header  |  1993-09-26  |  87KB  |  3,247 lines

  1. /****************************************************************************
  2. *                parse.c
  3. *
  4. *  This module implements a parser for the scene description files.
  5. *
  6. *  from Persistence of Vision Raytracer
  7. *  Copyright 1993 Persistence of Vision Team
  8. *---------------------------------------------------------------------------
  9. *  NOTICE: This source code file is provided so that users may experiment
  10. *  with enhancements to POV-Ray and to port the software to platforms other 
  11. *  than those supported by the POV-Ray Team.  There are strict rules under
  12. *  which you are permitted to use this file.  The rules are in the file
  13. *  named POVLEGAL.DOC which should be distributed with this file. If 
  14. *  POVLEGAL.DOC is not available or for more info please contact the POV-Ray
  15. *  Team Coordinator by leaving a message in CompuServe's Graphics Developer's
  16. *  Forum.  The latest version of POV-Ray may be found there as well.
  17. *
  18. * This program is based on the popular DKB raytracer version 2.12.
  19. * DKBTrace was originally written by David K. Buck.
  20. * DKBTrace Ver 2.0-2.12 were written by David K. Buck & Aaron A. Collins.
  21. *
  22. *****************************************************************************/
  23.  
  24. #include "frame.h"
  25. #include "vector.h"
  26. #include "povproto.h"
  27. #include "parse.h"
  28.  
  29. /* This file implements a simple recursive-descent parser for reading the
  30. input file.  */
  31.  
  32. extern DBL Max_Trace_Level;
  33. extern char VerboseFormat;
  34. extern unsigned int Options;
  35. extern int Use_Slabs;
  36. extern char Stat_File_Name[FILE_NAME_LENGTH];
  37.  
  38. extern struct Reserved_Word_Struct Reserved_Words [];
  39. extern DBL Antialias_Threshold;
  40.  
  41. extern struct Token_Struct Token;
  42. extern char String[MAX_STRING_INDEX];
  43.  
  44. extern COLOUR_MAP_ENTRY *Build_Entries;
  45. extern FRAME Frame;
  46. extern DBL Clock_Value;
  47. extern char **Symbol_Table;
  48. extern int Max_Intersections;
  49. extern DBL Language_Version;
  50. extern METHODS Csg_Height_Field_Methods;
  51. extern METHODS CSG_Union_Methods;
  52.  
  53. static void Parse_Image_Map PARAMS((PIGMENT *Pigment));
  54. static void Parse_Bump_Map PARAMS((TNORMAL *Tnormal));
  55. static void Parse_Pigment PARAMS((PIGMENT **Pigment_Ptr));
  56. static void Parse_Tnormal PARAMS((TNORMAL **Tnormal_Ptr));
  57. static void Parse_Finish PARAMS((FINISH **Finish_Ptr));
  58. static TEXTURE *Parse_Texture PARAMS((void));
  59. static void Token_Init PARAMS((void));
  60. static void Frame_Init PARAMS((void));
  61. static void Parse_Coeffs PARAMS((int order, DBL *Coeffs));
  62. static IMAGE *Parse_Image PARAMS((int Legal));
  63. static TRANSFORM *Parse_Transform PARAMS((void));
  64. static void Parse_Object_Mods PARAMS((OBJECT *Object));
  65. static OBJECT *Parse_Bicubic_Patch PARAMS((void));
  66. static OBJECT *Parse_Blob PARAMS((void));
  67. static OBJECT *Parse_Box PARAMS((void));
  68. static OBJECT *Parse_Cone PARAMS((void));
  69. static OBJECT *Parse_Cylinder PARAMS((void));
  70. static OBJECT *Parse_Disc PARAMS((void));
  71. static OBJECT *Parse_Height_Field PARAMS((void));
  72. static OBJECT *Parse_Plane PARAMS((void));
  73. static OBJECT *Parse_Poly PARAMS((int order));
  74. static OBJECT *Parse_Quadric PARAMS((void));
  75. static OBJECT *Parse_Smooth_Triangle PARAMS((void));
  76. static OBJECT *Parse_Sphere PARAMS((void));
  77. static OBJECT *Parse_Torus PARAMS((void));
  78. static OBJECT *Parse_Triangle PARAMS((void));
  79. static OBJECT *Parse_CSG PARAMS((int CSG_Type));
  80. static OBJECT *Parse_Light_Source PARAMS((void));
  81. static OBJECT *Parse_Object PARAMS((void));
  82. static void Parse_Fog PARAMS((void));
  83. static void Parse_Frame PARAMS((void));
  84. static void Parse_Camera PARAMS((CAMERA **Camera_Ptr));
  85. static void Parse_Declare PARAMS((void));
  86. static void Link PARAMS((OBJECT *New_Object,OBJECT **Field,OBJECT **Old_Object_List));
  87. static void Link_Textures PARAMS((TEXTURE **Old_Texture, TEXTURE *New_Texture));
  88. static char *Get_Token_String PARAMS((TOKEN Token_Id));
  89. static void Where_Error PARAMS((void));
  90. static int Test_Redefine PARAMS((int a));
  91. static OBJECT *Parse_Bound_Clip PARAMS((void));
  92. static void Found_Instead PARAMS((void));
  93. /*static void Parse_Warn PARAMS((TOKEN Token_Id));*/
  94. static void Warn_State PARAMS((TOKEN Token_Id, TOKEN Type));
  95. static void Post_Process PARAMS((OBJECT *Object, OBJECT *Parent));
  96. static void Destroy_Constants PARAMS((void));
  97. static OBJECT *Parse_Object_Id PARAMS((void));
  98. static void Link_To_Frame PARAMS((OBJECT *Object));
  99.  
  100. extern struct Constant_Struct Constants[MAX_CONSTANTS];
  101.  
  102. int Number_Of_Constants;
  103. int Previous;
  104. short Have_Vector;
  105. short Not_In_Default;
  106.  
  107. TOKEN *Brace_Stack;
  108. int Brace_Index;
  109.  
  110. TEXTURE *Default_Texture;
  111. CAMERA *Default_Camera;
  112.  
  113. /* Parse the file. */
  114. void Parse ()
  115.   {
  116.    Build_Entries  = NULL;
  117.    if ((Brace_Stack = (TOKEN *) malloc(MAX_BRACES*sizeof (TOKEN))) == NULL)
  118.      MAError ("brace stack");
  119.    Brace_Index = 0;
  120.    Token_Init ();
  121.    Default_Camera = Create_Camera();
  122.    Default_Texture = Create_PNF_Texture();
  123.    Default_Texture->Pigment = Create_Pigment();
  124.    Default_Texture->Tnormal = NULL;
  125.    Default_Texture->Finish  = Create_Finish();
  126.    Not_In_Default = TRUE;
  127.    Frame_Init ();
  128.    Parse_Frame ();
  129.    if (Frame.Objects==NULL)
  130.      Error("No objects in scene");
  131.    Destroy_Constants ();
  132.    Destroy_Textures(Default_Texture);
  133.    Destroy_Camera(Default_Camera);
  134.    if (Build_Entries != NULL)
  135.      free (Build_Entries);
  136.    free (Brace_Stack);
  137.   }
  138.  
  139. static void Token_Init ()
  140.   {
  141.    Number_Of_Constants = 0;
  142.   }
  143.  
  144. /* Set up the fields in the frame to default values. */
  145. static
  146. void Frame_Init ()
  147.   {
  148.    Frame.Camera = Copy_Camera(Default_Camera);
  149.    Frame.Light_Sources = NULL;
  150.    Frame.Objects = NULL;
  151.    Frame.Atmosphere_IOR = 1.0;
  152.    Frame.Antialias_Threshold = Antialias_Threshold;
  153.    Frame.Fog_Distance = 0.0;
  154.    Make_Colour (&(Frame.Fog_Colour), 0.0, 0.0, 0.0);
  155.   }
  156.  
  157. void Parse_Begin ()
  158.   {
  159.    char *front;
  160.  
  161.    Brace_Stack[++Brace_Index]=Token.Token_Id;
  162.    Get_Token ();
  163.  
  164.    if (Token.Token_Id == LEFT_CURLY_TOKEN)
  165.      return;
  166.  
  167.    Where_Error ();
  168.  
  169.    front = Get_Token_String (Brace_Stack[Brace_Index]);
  170.    fprintf (stderr, "Missing { after %s, ", front);
  171.    Found_Instead ();
  172.    exit (1);
  173.   }
  174.  
  175. void Parse_End ()
  176.   {
  177.    char *front;
  178.  
  179.    Get_Token ();
  180.  
  181.    if (Token.Token_Id == RIGHT_CURLY_TOKEN)
  182.      {
  183.       Brace_Index--;
  184.       return;
  185.      }
  186.  
  187.    Where_Error ();
  188.  
  189.    front = Get_Token_String (Brace_Stack[Brace_Index]);
  190.    fprintf (stderr, "No matching } in %s,", front);
  191.    Found_Instead ();
  192.    exit (1);
  193.   }
  194.  
  195. static OBJECT *Parse_Object_Id ()
  196.   {
  197.    OBJECT *Object;
  198.    
  199.    EXPECT
  200.      CASE (OBJECT_ID_TOKEN)
  201.        Warn_State(OBJECT_ID_TOKEN, OBJECT_TOKEN);
  202.        Object = Copy_Object((OBJECT *) Token.Constant_Data);
  203.        Parse_Object_Mods (Object);
  204.        EXIT
  205.      END_CASE
  206.  
  207.      OTHERWISE
  208.        Object = NULL;
  209.        UNGET
  210.        EXIT
  211.      END_CASE
  212.    END_EXPECT
  213.  
  214.    return (Object);
  215.   }
  216.  
  217. void Parse_Comma ()
  218.   {
  219.    Get_Token();
  220.    if (Token.Token_Id != COMMA_TOKEN)
  221.      {
  222.       UNGET;
  223.      }
  224.   }
  225.  
  226. static void Parse_Coeffs(order, Coeffs)
  227.   int order;
  228.   DBL *Coeffs;
  229.   {
  230.    int i;
  231.  
  232.    EXPECT
  233.      CASE (LEFT_ANGLE_TOKEN)
  234.        Coeffs[0] = Parse_Float();
  235.        for (i = 1; i < term_counts(order); i++)
  236.          {
  237.           Parse_Comma();
  238.           Coeffs[i] = Parse_Float();
  239.          }
  240.        GET (RIGHT_ANGLE_TOKEN);
  241.        EXIT
  242.      END_CASE
  243.  
  244.      OTHERWISE
  245.        Parse_Error (LEFT_ANGLE_TOKEN);
  246.      END_CASE
  247.    END_EXPECT
  248.   }
  249.  
  250. static
  251. IMAGE *Parse_Image (Legal)
  252.   int Legal;
  253.   {
  254.    IMAGE *Image;
  255.    VECTOR Local_Vector;
  256.  
  257.    Image = Create_Image ();
  258.  
  259.    if (Legal & GRAD_FILE)
  260.      {
  261.       EXPECT
  262.         CASE_VECTOR
  263.           Warn("Should use map_type keyword and/or eliminate orientation.",1.5);
  264.           Have_Vector = FALSE;
  265.           Parse_Vector_Float (&Local_Vector);
  266.           if (Have_Vector)
  267.             Image->Gradient = Local_Vector;
  268.           else
  269.             Image->Map_Type = (int)Local_Vector.x;
  270.         END_CASE
  271.  
  272.         OTHERWISE
  273.           UNGET
  274.           EXIT
  275.         END_CASE
  276.       END_EXPECT
  277.      }
  278.  
  279.    EXPECT
  280.      CASE (IFF_TOKEN)
  281.        Image->File_Type = IFF_FILE;
  282.        GET (STRING_TOKEN);
  283.        Read_Iff_Image (Image, Token.Token_String);
  284.        EXIT
  285.      END_CASE
  286.  
  287.      CASE (GIF_TOKEN)
  288.        Image->File_Type = GIF_FILE;
  289.        GET (STRING_TOKEN);
  290.        Read_Gif_Image(Image, Token.Token_String);
  291.        EXIT
  292.      END_CASE
  293.  
  294.      CASE (POT_TOKEN)
  295.        Image->File_Type = POT_FILE;
  296.        GET (STRING_TOKEN);
  297.        Read_Gif_Image(Image, Token.Token_String);
  298.        EXIT
  299.      END_CASE
  300.  
  301.      CASE (DUMP_TOKEN)
  302.        Image->File_Type = DUMP_FILE;
  303.        GET (STRING_TOKEN);
  304.        Read_Dump_Image(Image, Token.Token_String);
  305.        EXIT
  306.      END_CASE
  307.  
  308.      CASE (TGA_TOKEN)
  309.        Image->File_Type = TGA_FILE;
  310.        GET (STRING_TOKEN);
  311.        Read_Targa_Image(Image, Token.Token_String);
  312.        EXIT
  313.      END_CASE
  314.  
  315.      OTHERWISE
  316.        Parse_Error_Str ("map file spec");
  317.      END_CASE
  318.    END_EXPECT
  319.  
  320.    if (!(Image->File_Type & Legal))
  321.      Error ("File type not supported here");
  322.    return (Image);
  323.   }
  324.  
  325. static void Parse_Image_Map (Pigment)
  326.   PIGMENT *Pigment;
  327.   {
  328.    int reg;
  329.  
  330.    Pigment->Type = IMAGE_MAP_PIGMENT;
  331.  
  332.    Parse_Begin ();
  333.  
  334.    Pigment->Image = Parse_Image (IMAGE_FILE);
  335.    Pigment->Image->Use_Colour_Flag = TRUE;
  336.  
  337.    EXPECT                   /* Look for image_attribs */
  338.      CASE (ONCE_TOKEN)
  339.        Pigment->Image->Once_Flag=TRUE;
  340.      END_CASE
  341.  
  342.      CASE (INTERPOLATE_TOKEN)
  343.        Pigment->Image->Interpolation_Type = (int)Parse_Float();
  344.      END_CASE
  345.  
  346.      CASE (MAP_TYPE_TOKEN)
  347.        Pigment->Image->Map_Type = (int) Parse_Float ();
  348.      END_CASE
  349.  
  350.      CASE (USE_COLOUR_TOKEN)
  351.        Pigment->Image->Use_Colour_Flag = TRUE;
  352.      END_CASE
  353.  
  354.      CASE (USE_INDEX_TOKEN)
  355.        Pigment->Image->Use_Colour_Flag = FALSE;
  356.      END_CASE
  357.  
  358.      CASE (ALPHA_TOKEN)
  359.        Warn("Keyword ALPHA discontinued.  Use FILTER instead.",1.55);
  360.          
  361.      CASE (FILTER_TOKEN)
  362.        EXPECT
  363.          CASE (ALL_TOKEN)
  364.            {
  365.             DBL filter;
  366.             filter = Parse_Float();
  367.             for (reg = 0 ; reg < Pigment->Image->Colour_Map_Size ; reg++)
  368.               Pigment->Image->Colour_Map[reg].Filter
  369.                   = (unsigned short) (filter *255.0);
  370.            }
  371.            EXIT
  372.          END_CASE
  373.  
  374.          OTHERWISE
  375.            UNGET
  376.            reg = (int)(Parse_Float() + 0.01);
  377.            if (Pigment->Image->Colour_Map == NULL)
  378.              Error ("Can't apply FILTER to a non colour-mapped image\n");
  379.            if ((reg < 0) || (reg >= Pigment->Image->Colour_Map_Size))
  380.              Error ("FILTER colour register value out of range.\n");
  381.  
  382.            Parse_Comma();
  383.            Pigment->Image->Colour_Map[reg].Filter
  384.                   = (unsigned short) (255.0 * Parse_Float());
  385.            EXIT
  386.          END_CASE
  387.  
  388.        END_EXPECT
  389.      END_CASE
  390.  
  391.      OTHERWISE
  392.        UNGET
  393.        EXIT
  394.      END_CASE
  395.    END_EXPECT
  396.  
  397.    Parse_End ();
  398. }
  399.  
  400. static void Parse_Bump_Map (Tnormal)
  401.   TNORMAL *Tnormal;
  402.   {
  403.    Tnormal->Type = BUMP_MAP;
  404.  
  405.    Parse_Begin ();
  406.  
  407.    Tnormal->Image = Parse_Image (NORMAL_FILE);
  408.  
  409.    Tnormal->Image->Use_Colour_Flag = TRUE;
  410.  
  411.    EXPECT
  412.      CASE (ONCE_TOKEN)
  413.        Tnormal->Image->Once_Flag=TRUE;
  414.      END_CASE
  415.  
  416.      CASE (MAP_TYPE_TOKEN)
  417.        Tnormal->Image->Map_Type = (int) Parse_Float ();
  418.      END_CASE
  419.  
  420.      CASE (INTERPOLATE_TOKEN)
  421.        Tnormal->Image->Interpolation_Type = (int)Parse_Float();
  422.      END_CASE
  423.  
  424.      CASE (BUMP_SIZE_TOKEN)
  425.        Tnormal->Amount = Parse_Float ();
  426.      END_CASE
  427.  
  428.      CASE (USE_COLOUR_TOKEN)
  429.        Tnormal->Image->Use_Colour_Flag = TRUE;
  430.      END_CASE
  431.  
  432.      CASE (USE_INDEX_TOKEN)
  433.        Tnormal->Image->Use_Colour_Flag = FALSE;
  434.      END_CASE
  435.  
  436.      OTHERWISE
  437.        UNGET
  438.        EXIT
  439.      END_CASE
  440.    END_EXPECT
  441.    Parse_End ();
  442. }
  443.  
  444. static void Parse_Pigment (Pigment_Ptr)
  445.   PIGMENT **Pigment_Ptr;
  446.   {
  447.    PIGMENT *New;
  448.    VECTOR Local_Vector;
  449.  
  450.    Parse_Begin ();
  451.  
  452.    EXPECT            /* Look for [pigment_id] */
  453.      CASE (PIGMENT_ID_TOKEN)
  454.        Destroy_Pigment(*Pigment_Ptr);
  455.        *Pigment_Ptr = Copy_Pigment ((PIGMENT *) Token.Constant_Data);
  456.        EXIT
  457.      END_CASE
  458.  
  459.      OTHERWISE
  460.        UNGET
  461.        EXIT
  462.      END_CASE
  463.    END_EXPECT    /* End pigment_id */
  464.  
  465.    New = *Pigment_Ptr;
  466.  
  467.    EXPECT
  468.      CASE (AGATE_TOKEN)
  469.        New->Type = AGATE_PIGMENT;
  470.        EXIT
  471.      END_CASE
  472.  
  473.      CASE (BOZO_TOKEN)
  474.        New->Type = BOZO_PIGMENT;
  475.        EXIT
  476.      END_CASE
  477.  
  478.      CASE (GRANITE_TOKEN)
  479.        New->Type = GRANITE_PIGMENT;
  480.        EXIT
  481.      END_CASE
  482.  
  483.      CASE (LEOPARD_TOKEN)
  484.        New->Type = LEOPARD_PIGMENT;
  485.        EXIT
  486.      END_CASE
  487.  
  488.      CASE (MARBLE_TOKEN)
  489.        New->Type = MARBLE_PIGMENT;
  490.        EXIT
  491.      END_CASE
  492.  
  493.      CASE (MANDEL_TOKEN)
  494.        New->Type = MANDEL_PIGMENT;
  495.        New->Iterations = (int)Parse_Float();
  496.        EXIT
  497.      END_CASE
  498.  
  499.      CASE (ONION_TOKEN)
  500.        New->Type = ONION_PIGMENT;
  501.        EXIT
  502.      END_CASE
  503.  
  504.      CASE (PAINTED1_TOKEN)
  505.        New->Type = PAINTED1_PIGMENT;
  506.        EXIT
  507.      END_CASE
  508.  
  509.      CASE (PAINTED2_TOKEN)
  510.        New->Type = PAINTED2_PIGMENT;
  511.        EXIT
  512.      END_CASE
  513.  
  514.      CASE (PAINTED3_TOKEN)
  515.        New->Type = PAINTED2_PIGMENT;
  516.        EXIT
  517.      END_CASE
  518.  
  519.      CASE (SPOTTED_TOKEN)
  520.        New->Type = SPOTTED_PIGMENT;
  521.        EXIT
  522.      END_CASE
  523.  
  524.      CASE (WOOD_TOKEN)
  525.        New->Type = WOOD_PIGMENT;
  526.        EXIT
  527.      END_CASE
  528.  
  529.      CASE (GRADIENT_TOKEN)
  530.        New->Type = GRADIENT_PIGMENT;
  531.        Parse_Vector (&(New->Colour_Gradient));
  532.        EXIT
  533.      END_CASE
  534.  
  535.      CASE (RADIAL_TOKEN)
  536.        New->Type = RADIAL_PIGMENT;
  537.      END_CASE
  538.  
  539.      CASE (COLOUR_TOKEN)
  540.        New->Type = COLOUR_PIGMENT;
  541.        New->Colour1 = Create_Colour ();
  542.        Parse_Colour (New->Colour1);
  543.        New->Quick_Colour = *New->Colour1;
  544.        EXIT
  545.      END_CASE
  546.  
  547.      CASE5 (COLOUR_ID_TOKEN, RGB_TOKEN, RGBF_TOKEN, RED_TOKEN, BLUE_TOKEN)
  548.      CASE3 (GREEN_TOKEN, ALPHA_TOKEN, FILTER_TOKEN)
  549.        UNGET
  550.        New->Type = COLOUR_PIGMENT;
  551.        New->Colour1 = Create_Colour ();
  552.        Parse_Colour (New->Colour1);
  553.        New->Quick_Colour = *New->Colour1;
  554.        EXIT
  555.      END_CASE
  556.  
  557.      CASE (CHECKER_TOKEN)
  558.        New->Type = CHECKER_PIGMENT;
  559.        New->Colour_Map = Parse_Colour_List(2);
  560.        EXIT
  561.      END_CASE
  562.  
  563.      CASE (HEXAGON_TOKEN)
  564.        New->Type = HEXAGON_PIGMENT;
  565.        New->Colour_Map = Parse_Colour_List(3);
  566.        EXIT
  567.      END_CASE
  568.  
  569.      CASE (IMAGE_MAP_TOKEN)
  570.        Parse_Image_Map (New);
  571.        EXIT
  572.      END_CASE
  573.  
  574.      OTHERWISE
  575.        UNGET
  576.        EXIT
  577.      END_CASE
  578.    END_EXPECT     /* Concludes pigment_body */
  579.  
  580.    EXPECT         /* Look for pigment_modifier */
  581.      CASE (TURBULENCE_TOKEN)
  582.        Parse_Vector_Float(&(New->Turbulence));
  583.        if ((New->Turbulence.x !=0.0) || (New->Turbulence.y !=0.0) ||
  584.            (New->Turbulence.z !=0.0))
  585.           New->Flags |= HAS_TURB;
  586.      END_CASE
  587.  
  588.      CASE (COLOUR_MAP_TOKEN)
  589.        if (New->Type == CHECKER_PIGMENT ||
  590.            New->Type == HEXAGON_PIGMENT ||
  591.            New->Type == COLOUR_PIGMENT ||
  592.            New->Type == IMAGE_MAP_PIGMENT)
  593.          Warn ("Cannot use color map with this pigment type",1.5);
  594.        New->Colour_Map = Parse_Colour_Map ();
  595.      END_CASE
  596.  
  597.      CASE (QUICK_COLOUR_TOKEN)
  598.        Parse_Colour (&New->Quick_Colour);
  599.      END_CASE
  600.  
  601.      CASE (OCTAVES_TOKEN)
  602.        New->Octaves = (int)Parse_Float();
  603.          if(New->Octaves < 1)
  604.             New->Octaves = 1;
  605.          if(New->Octaves > 10)  /* Avoid DOMAIN errors */
  606.             New->Octaves = 10;
  607.      END_CASE
  608.  
  609.      CASE (OMEGA_TOKEN)
  610.        New->omega = Parse_Float();
  611.      END_CASE
  612.  
  613.      CASE (LAMBDA_TOKEN)
  614.        New->lambda = Parse_Float();
  615.      END_CASE
  616.  
  617.      CASE (FREQUENCY_TOKEN)
  618.        New->Frequency = Parse_Float();
  619.      END_CASE
  620.  
  621.      CASE (PHASE_TOKEN)
  622.        New->Phase = Parse_Float();
  623.      END_CASE
  624.  
  625.      CASE (AGATE_TURB_TOKEN)
  626.        if (Not_In_Default && (New->Type != AGATE_PIGMENT))
  627.           Warn("Attempt to use agate_turb on non-agate",1.9);
  628.        New->Agate_Turb_Scale = Parse_Float();
  629.      END_CASE
  630.  
  631.      CASE (TRANSLATE_TOKEN)
  632.        Parse_Vector (&Local_Vector);
  633.        Translate_Pigment (New, &Local_Vector);
  634.      END_CASE
  635.  
  636.      CASE (ROTATE_TOKEN)
  637.        Parse_Vector (&Local_Vector);
  638.        Rotate_Pigment (New, &Local_Vector);
  639.      END_CASE
  640.  
  641.      CASE (SCALE_TOKEN)
  642.        Parse_Scale_Vector (&Local_Vector);
  643.        Scale_Pigment (New, &Local_Vector);
  644.      END_CASE
  645.  
  646.      CASE (TRANSFORM_TOKEN)
  647.        GET(TRANSFORM_ID_TOKEN)
  648.        Transform_Pigment (New, (TRANSFORM *)Token.Constant_Data);
  649.      END_CASE
  650.  
  651.      OTHERWISE
  652.        UNGET
  653.        EXIT
  654.      END_CASE
  655.    END_EXPECT
  656.  
  657.    if (Not_In_Default && (New->Type == NO_PIGMENT))
  658.      Warn("Pigment type unspecified or not 1st item",1.7);
  659.  
  660.    Parse_End ();
  661.   }
  662.  
  663. static void Parse_Tnormal (Tnormal_Ptr)
  664.   TNORMAL **Tnormal_Ptr;
  665.   {
  666.    TNORMAL *New;
  667.    VECTOR Local_Vector;
  668.  
  669.    Parse_Begin ();
  670.  
  671.    EXPECT            /* Look for [tnormal_id] */
  672.      CASE (TNORMAL_ID_TOKEN)
  673.        Destroy_Tnormal(*Tnormal_Ptr);
  674.        *Tnormal_Ptr = Copy_Tnormal ((TNORMAL *) Token.Constant_Data);
  675.        EXIT
  676.      END_CASE
  677.  
  678.      OTHERWISE
  679.        UNGET
  680.        EXIT
  681.      END_CASE
  682.    END_EXPECT    /* End [tnormal_id] */
  683.  
  684.    if (*Tnormal_Ptr == NULL)
  685.      if ((Default_Texture->Tnormal) != NULL)
  686.        *Tnormal_Ptr = Copy_Tnormal ((Default_Texture->Tnormal));
  687.      else
  688.        *Tnormal_Ptr = Create_Tnormal ();
  689.  
  690.    New = *Tnormal_Ptr;
  691.  
  692.    EXPECT  /* [tnormal_body] */
  693.      CASE (BUMPS_TOKEN)
  694.        New->Type = BUMPS;
  695.        New->Amount = Parse_Float ();
  696.        EXIT
  697.      END_CASE
  698.  
  699.      CASE (BUMPY1_TOKEN)
  700.        New->Type = BUMPY1;
  701.        New->Amount = Parse_Float ();
  702.        EXIT
  703.      END_CASE
  704.  
  705.      CASE (BUMPY2_TOKEN)
  706.        New->Type = BUMPY2;
  707.        New->Amount = Parse_Float ();
  708.        EXIT
  709.      END_CASE
  710.  
  711.      CASE (BUMPY3_TOKEN)
  712.        New->Type = BUMPY3;
  713.        New->Amount = Parse_Float ();
  714.        EXIT
  715.      END_CASE
  716.  
  717.      CASE (DENTS_TOKEN)
  718.        New->Type = DENTS;
  719.        New->Amount = Parse_Float ();
  720.        EXIT
  721.      END_CASE
  722.  
  723.      CASE (RIPPLES_TOKEN)
  724.        New->Type = RIPPLES;
  725.        New->Amount = Parse_Float ();
  726.        EXIT
  727.      END_CASE
  728.  
  729.      CASE (WAVES_TOKEN)
  730.        New->Type = WAVES;
  731.        New->Amount = Parse_Float ();
  732.        EXIT
  733.      END_CASE
  734.  
  735.      CASE (WRINKLES_TOKEN)
  736.        New->Type = WRINKLES;
  737.        New->Amount = Parse_Float ();
  738.        EXIT
  739.      END_CASE
  740.  
  741.      CASE (BUMP_MAP_TOKEN)
  742.        Parse_Bump_Map (New);
  743.        EXIT
  744.      END_CASE
  745.  
  746.      OTHERWISE
  747.        if (Not_In_Default && (New->Type == NO_NORMAL))
  748.          Parse_Error_Str("normal body");
  749.        UNGET
  750.        EXIT
  751.      END_CASE
  752.    END_EXPECT    /* End of tnormal_body */
  753.  
  754.    EXPECT        /* Look for tnormal_mods */
  755.  
  756.      CASE (TURBULENCE_TOKEN)
  757.        Parse_Vector_Float(&(New->Turbulence));
  758.        if ((New->Turbulence.x !=0.0) || (New->Turbulence.y !=0.0) ||
  759.            (New->Turbulence.z !=0.0))
  760.           New->Flags |= HAS_TURB;
  761.      END_CASE
  762.  
  763.      CASE (OCTAVES_TOKEN)
  764.        New->Octaves = (int)Parse_Float();
  765.      END_CASE
  766.  
  767.      CASE (OMEGA_TOKEN)
  768.        New->omega = Parse_Float();
  769.      END_CASE
  770.  
  771.      CASE (LAMBDA_TOKEN)
  772.        New->lambda = Parse_Float();
  773.      END_CASE
  774.  
  775.      CASE (FREQUENCY_TOKEN)
  776.        if (!(New->Type == RIPPLES || New->Type == WAVES))
  777.          if (Language_Version >= 1.5)
  778.            Warn ("Cannot use frequency with this normal",1.5);
  779.        New->Frequency = Parse_Float();
  780.      END_CASE
  781.  
  782.      CASE (PHASE_TOKEN)
  783.        if (!(New->Type == RIPPLES || New->Type == WAVES))
  784.          if (Language_Version >= 1.5)
  785.             Warn ("Cannot use phase with this normal",1.5);
  786.        New->Phase = Parse_Float();
  787.      END_CASE
  788.  
  789.      CASE (TRANSLATE_TOKEN)
  790.        Parse_Vector (&Local_Vector);
  791.        Translate_Tnormal (New, &Local_Vector);
  792.      END_CASE
  793.  
  794.      CASE (ROTATE_TOKEN)
  795.        Parse_Vector (&Local_Vector);
  796.        Rotate_Tnormal (New, &Local_Vector);
  797.      END_CASE
  798.  
  799.      CASE (SCALE_TOKEN)
  800.        Parse_Scale_Vector (&Local_Vector);
  801.        Scale_Tnormal (New, &Local_Vector);
  802.      END_CASE
  803.  
  804.      CASE (TRANSFORM_TOKEN)
  805.        GET(TRANSFORM_ID_TOKEN)
  806.        Transform_Tnormal (New, (TRANSFORM *)Token.Constant_Data);
  807.      END_CASE
  808.  
  809.      OTHERWISE
  810.        UNGET
  811.        EXIT
  812.      END_CASE
  813.    END_EXPECT    /* End of tnormal_mods */
  814.  
  815.    Parse_End ();
  816.   }
  817.  
  818. static void Parse_Finish (Finish_Ptr)
  819.   FINISH **Finish_Ptr;
  820.   {
  821.    FINISH *New;
  822.  
  823.    Parse_Begin ();
  824.  
  825.    EXPECT        /* Look for zero or one finish_id */
  826.      CASE (FINISH_ID_TOKEN)
  827.        Destroy_Finish(*Finish_Ptr);
  828.        *Finish_Ptr = Copy_Finish ((FINISH *) Token.Constant_Data);
  829.        EXIT
  830.      END_CASE
  831.  
  832.      OTHERWISE
  833.        UNGET
  834.        EXIT
  835.      END_CASE
  836.    END_EXPECT    /* End finish_id */
  837.  
  838.    New = *Finish_Ptr;
  839.  
  840.    EXPECT        /* Look for zero or more finish_body */
  841.      CASE (AMBIENT_TOKEN)
  842.        New->Ambient = Parse_Float ();
  843.      END_CASE
  844.  
  845.      CASE (BRILLIANCE_TOKEN)
  846.        New->Brilliance = Parse_Float ();
  847.      END_CASE
  848.  
  849.      CASE (DIFFUSE_TOKEN)
  850.        New->Diffuse = Parse_Float ();
  851.      END_CASE
  852.  
  853.      CASE (REFLECTION_TOKEN)
  854.        New->Reflection = Parse_Float ();
  855.      END_CASE
  856.  
  857.      CASE (REFRACTION_TOKEN)
  858.        New->Refraction = Parse_Float ();
  859.      END_CASE
  860.  
  861.      CASE (IOR_TOKEN)
  862.        New->Index_Of_Refraction = Parse_Float ();
  863.      END_CASE
  864.  
  865.      CASE (PHONG_TOKEN)
  866.        New->Phong = Parse_Float ();
  867.      END_CASE
  868.  
  869.      CASE (PHONG_SIZE_TOKEN)
  870.        New->Phong_Size = Parse_Float ();
  871. /*     if (New->Phong_Size < 1.0)
  872.            New->Phong_Size = 1.0;
  873.        if (New->Phong_Size > 100)
  874.            New->Phong_Size = 100; */
  875.      END_CASE
  876.  
  877.      CASE (SPECULAR_TOKEN)
  878.        New->Specular = Parse_Float ();
  879.      END_CASE
  880.  
  881.      CASE (ROUGHNESS_TOKEN)
  882.        New->Roughness = Parse_Float ();
  883. /*     if (New->Roughness > 1.0)
  884.            New->Roughness = 1.0;
  885.        if (New->Roughness < 0.001)
  886.            New->Roughness = 0.001;  */
  887.        New->Roughness = 1.0/New->Roughness; /* CEY 12/92 */
  888.      END_CASE
  889.  
  890.      CASE (METALLIC_TOKEN)
  891.        New->Metallic_Flag = TRUE;
  892.      END_CASE
  893.  
  894.      CASE (CRAND_TOKEN)
  895.        New->Crand = Parse_Float();
  896.      END_CASE
  897.  
  898.      OTHERWISE
  899.        UNGET
  900.        EXIT
  901.      END_CASE
  902.    END_EXPECT    /* End of finish_body */
  903.  
  904.    EXPECT        /* Look for finish_mods */
  905.  
  906. /*   CASE none implemented
  907.      END_CASE     */
  908.  
  909.      OTHERWISE
  910.        UNGET
  911.        EXIT
  912.      END_CASE
  913.    END_EXPECT    /* End of finish_mods */
  914.  
  915.    Parse_End ();
  916.   }
  917.  
  918. #define ADD_TNORMAL if (Tnormal == NULL) {if ((Default_Texture->Tnormal) != NULL) \
  919.  Tnormal = Copy_Tnormal ((Default_Texture->Tnormal)); else Tnormal = Create_Tnormal ();\
  920.  Texture->Tnormal=Tnormal;};
  921.  
  922. static
  923. TEXTURE *Parse_Texture ()
  924.   {
  925.    VECTOR Local_Vector;
  926.    TEXTURE *Texture, *Local_Texture;
  927.    PIGMENT *Pigment;
  928.    TNORMAL *Tnormal;
  929.    FINISH *Finish;
  930.  
  931.    Parse_Begin ();
  932.  
  933.    EXPECT                      /* Look for texture_body */
  934.      CASE (TILES_TOKEN)
  935.        Parse_Begin ();
  936.  
  937.        Texture = (TEXTURE *)Create_Tiles_Texture ();
  938.  
  939.        EXPECT
  940.          CASE (TEXTURE_TOKEN)
  941.            Local_Texture = Parse_Texture ();
  942.            Link_Textures(&(((TILES *)Texture)->Tile1),Local_Texture);
  943.          END_CASE
  944.  
  945.          OTHERWISE
  946.            UNGET
  947.            EXIT
  948.          END_CASE
  949.        END_EXPECT
  950.  
  951.        GET (TILE2_TOKEN);
  952.  
  953.        EXPECT
  954.          CASE (TEXTURE_TOKEN)
  955.            Local_Texture = Parse_Texture ();
  956.            Link_Textures(&(((TILES *)Texture)->Tile2),Local_Texture);
  957.          END_CASE
  958.  
  959.          OTHERWISE
  960.            UNGET
  961.            EXIT
  962.          END_CASE
  963.        END_EXPECT
  964.        Parse_End ();
  965.        EXIT
  966.      END_CASE
  967.  
  968.      CASE (MATERIAL_MAP_TOKEN)
  969.        Parse_Begin ();
  970.  
  971.        Texture = (TEXTURE *)Create_Material_Texture ();
  972.  
  973.        ((MATERIAL *)Texture)->Image = Parse_Image(MATERIAL_FILE);
  974.        ((MATERIAL *)Texture)->Image->Use_Colour_Flag = FALSE;
  975.  
  976.        EXPECT
  977.          CASE (ONCE_TOKEN)
  978.            ((MATERIAL *)Texture)->Image->Once_Flag=TRUE;
  979.          END_CASE
  980.  
  981.          CASE (INTERPOLATE_TOKEN)
  982.            ((MATERIAL *)Texture)->Image->Interpolation_Type=(int)Parse_Float();
  983.          END_CASE
  984.  
  985.          CASE (MAP_TYPE_TOKEN)
  986.            ((MATERIAL *)Texture)->Image->Map_Type = (int) Parse_Float ();
  987.          END_CASE
  988.  
  989.          OTHERWISE
  990.            UNGET
  991.            EXIT
  992.          END_CASE
  993.        END_EXPECT
  994.  
  995.        GET (TEXTURE_TOKEN)                /* First material */
  996.        ((MATERIAL *)Texture)->Materials = Local_Texture = Parse_Texture ();
  997.        ((MATERIAL *)Texture)->Num_Of_Mats++;
  998.  
  999.        EXPECT                             /* Subsequent materials */
  1000.          CASE (TEXTURE_TOKEN)
  1001.            Local_Texture->Next_Material = Parse_Texture ();
  1002.            Local_Texture = Local_Texture->Next_Material;
  1003.            ((MATERIAL *)Texture)->Num_Of_Mats++;
  1004.          END_CASE
  1005.  
  1006.          OTHERWISE
  1007.            UNGET
  1008.            EXIT
  1009.          END_CASE
  1010.        END_EXPECT
  1011.        Parse_End ();
  1012.        EXIT
  1013.      END_CASE
  1014.  
  1015.      OTHERWISE  /* Look for [pnf_texture] */
  1016.        UNGET
  1017.  
  1018.        Texture = Copy_Textures (Default_Texture);
  1019.  
  1020.        EXPECT   /* Look for [tpnf_ids] */
  1021.          CASE (TEXTURE_ID_TOKEN)
  1022.            Destroy_Textures(Texture);
  1023.            Texture = Copy_Textures((TEXTURE *) Token.Constant_Data);
  1024.          END_CASE
  1025.  
  1026.          CASE (PIGMENT_ID_TOKEN)
  1027.            Destroy_Pigment(Texture->Pigment);
  1028.            Texture->Pigment = Copy_Pigment ((PIGMENT *) Token.Constant_Data);
  1029.          END_CASE
  1030.  
  1031.          CASE (TNORMAL_ID_TOKEN)
  1032.            Destroy_Tnormal(Texture->Tnormal);
  1033.            Texture->Tnormal = Copy_Tnormal ((TNORMAL *) Token.Constant_Data);
  1034.          END_CASE
  1035.  
  1036.          CASE (FINISH_ID_TOKEN)
  1037.            Destroy_Finish(Texture->Finish);
  1038.            Texture->Finish = Copy_Finish ((FINISH *) Token.Constant_Data);
  1039.          END_CASE
  1040.  
  1041.          OTHERWISE
  1042.            UNGET
  1043.            EXIT
  1044.          END_CASE
  1045.        END_EXPECT
  1046.  
  1047.        Pigment = Texture->Pigment;
  1048.        Tnormal = Texture->Tnormal;
  1049.        Finish  = Texture->Finish;
  1050.  
  1051.        EXPECT
  1052.          CASE (PIGMENT_TOKEN)
  1053.            Parse_Pigment ( &(Texture->Pigment) );
  1054.          END_CASE
  1055.  
  1056.          CASE (TNORMAL_TOKEN)
  1057.            Parse_Tnormal ( &(Texture->Tnormal) );
  1058.          END_CASE
  1059.  
  1060.          CASE (FINISH_TOKEN)
  1061.            Parse_Finish ( &(Texture->Finish) );
  1062.          END_CASE
  1063.  
  1064. /***********************************************************************
  1065.  PIGMENT STUFF OUTSIDE PIGMENT{}
  1066. ***********************************************************************/
  1067.          CASE (AGATE_TOKEN)
  1068.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1069.            Pigment->Type = AGATE_PIGMENT;
  1070.          END_CASE
  1071.  
  1072.          CASE (BOZO_TOKEN)
  1073.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1074.            Pigment->Type = BOZO_PIGMENT;
  1075.          END_CASE
  1076.  
  1077.          CASE (GRANITE_TOKEN)
  1078.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1079.            Pigment->Type = GRANITE_PIGMENT;
  1080.          END_CASE
  1081.  
  1082.          CASE (LEOPARD_TOKEN)
  1083.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1084.            Pigment->Type = LEOPARD_PIGMENT;
  1085.          END_CASE
  1086.  
  1087.          CASE (MARBLE_TOKEN)
  1088.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1089.            Pigment->Type = MARBLE_PIGMENT;
  1090.          END_CASE
  1091.  
  1092.          CASE (MANDEL_TOKEN)
  1093.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1094.            Pigment->Type = MANDEL_PIGMENT;
  1095.            Pigment->Iterations = (int)Parse_Float();
  1096.          END_CASE
  1097.  
  1098.          CASE (ONION_TOKEN)
  1099.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1100.            Pigment->Type = ONION_PIGMENT;
  1101.          END_CASE
  1102.  
  1103.          CASE (PAINTED1_TOKEN)
  1104.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1105.            Pigment->Type = PAINTED1_PIGMENT;
  1106.          END_CASE
  1107.  
  1108.          CASE (PAINTED2_TOKEN)
  1109.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1110.            Pigment->Type = PAINTED2_PIGMENT;
  1111.          END_CASE
  1112.  
  1113.          CASE (PAINTED3_TOKEN)
  1114.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1115.            Pigment->Type = PAINTED2_PIGMENT;
  1116.          END_CASE
  1117.  
  1118.          CASE (SPOTTED_TOKEN)
  1119.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1120.            Pigment->Type = SPOTTED_PIGMENT;
  1121.          END_CASE
  1122.  
  1123.          CASE (WOOD_TOKEN)
  1124.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1125.            Pigment->Type = WOOD_PIGMENT;
  1126.          END_CASE
  1127.  
  1128.          CASE (GRADIENT_TOKEN)
  1129.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1130.            Pigment->Type = GRADIENT_PIGMENT;
  1131.            Parse_Vector (&(Pigment->Colour_Gradient));
  1132.          END_CASE
  1133.  
  1134.          CASE (COLOUR_TOKEN)
  1135.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1136.            Pigment->Type = COLOUR_PIGMENT;
  1137.            Pigment->Colour1 = Create_Colour ();
  1138.            Parse_Colour (Pigment->Colour1);
  1139.            Pigment->Quick_Colour = *Pigment->Colour1;
  1140.          END_CASE
  1141.  
  1142.          CASE5 (COLOUR_ID_TOKEN, RGB_TOKEN, RGBF_TOKEN, RED_TOKEN, BLUE_TOKEN)
  1143.          CASE3 (GREEN_TOKEN, ALPHA_TOKEN, FILTER_TOKEN)
  1144.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1145.            UNGET
  1146.            Pigment->Type = COLOUR_PIGMENT;
  1147.            Pigment->Colour1 = Create_Colour ();
  1148.            Parse_Colour (Pigment->Colour1);
  1149.            Pigment->Quick_Colour = *Pigment->Colour1;
  1150.          END_CASE
  1151.  
  1152.          CASE (CHECKER_TOKEN)
  1153.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1154.            Pigment->Type = CHECKER_PIGMENT;
  1155.            Pigment->Colour_Map = Parse_Colour_List(2);
  1156.          END_CASE
  1157.  
  1158.          CASE (HEXAGON_TOKEN)
  1159.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1160.            Pigment->Type = HEXAGON_PIGMENT;
  1161.            Pigment->Colour_Map = Parse_Colour_List(3);
  1162.          END_CASE
  1163.  
  1164.          CASE (IMAGE_MAP_TOKEN)
  1165.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1166.            Parse_Image_Map (Pigment);
  1167.          END_CASE
  1168.  
  1169.          CASE (TURBULENCE_TOKEN)
  1170.            Parse_Vector_Float(&(Pigment->Turbulence));
  1171.            if ((Pigment->Turbulence.x !=0.0) ||
  1172.                (Pigment->Turbulence.y !=0.0) ||
  1173.                (Pigment->Turbulence.z !=0.0))
  1174.              Pigment->Flags |= HAS_TURB;
  1175.          END_CASE
  1176.  
  1177.          CASE (COLOUR_MAP_TOKEN)
  1178.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);
  1179.            if (Pigment->Type == CHECKER_PIGMENT ||
  1180.                Pigment->Type == HEXAGON_PIGMENT ||
  1181.                Pigment->Type == COLOUR_PIGMENT ||
  1182.                Pigment->Type == IMAGE_MAP_PIGMENT)
  1183.              Warn ("Cannot use color map with this pigment type",1.5);
  1184.            Pigment->Colour_Map = Parse_Colour_Map ();
  1185.          END_CASE
  1186.  
  1187.          CASE (QUICK_COLOUR_TOKEN)
  1188.            Warn_State(Token.Token_Id, PIGMENT_TOKEN);           
  1189.            Parse_Colour (&Pigment->Quick_Colour);
  1190.          END_CASE
  1191.  
  1192.          CASE (OCTAVES_TOKEN)
  1193.            Pigment->Octaves = (int)Parse_Float();
  1194.              if(Pigment->Octaves < 1)
  1195.                 Pigment->Octaves = 1;
  1196.              if(Pigment->Octaves > 10)  /* Avoid DOMAIN errors */
  1197.                 Pigment->Octaves = 10;
  1198.          END_CASE
  1199.  
  1200.          CASE (OMEGA_TOKEN)
  1201.            Pigment->omega = Parse_Float();
  1202.          END_CASE
  1203.  
  1204.          CASE (LAMBDA_TOKEN)
  1205.            Pigment->lambda = Parse_Float();
  1206.          END_CASE
  1207.  
  1208. /***********************************************************************
  1209. TNORMAL STUFF OUTSIDE NORMAL{}
  1210. ***********************************************************************/
  1211.          CASE (BUMPS_TOKEN)
  1212.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1213.            ADD_TNORMAL
  1214.            Tnormal->Type = BUMPS;
  1215.            Tnormal->Amount = Parse_Float ();
  1216.          END_CASE
  1217.  
  1218.          CASE (BUMPY1_TOKEN)
  1219.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1220.            ADD_TNORMAL
  1221.            Tnormal->Type = BUMPY1;
  1222.            Tnormal->Amount = Parse_Float ();
  1223.          END_CASE
  1224.  
  1225.          CASE (BUMPY2_TOKEN)
  1226.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1227.            ADD_TNORMAL
  1228.            Tnormal->Type = BUMPY2;
  1229.            Tnormal->Amount = Parse_Float ();
  1230.          END_CASE
  1231.  
  1232.          CASE (BUMPY3_TOKEN)
  1233.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1234.            ADD_TNORMAL
  1235.            Tnormal->Type = BUMPY3;
  1236.            Tnormal->Amount = Parse_Float ();
  1237.          END_CASE
  1238.  
  1239.          CASE (DENTS_TOKEN)
  1240.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1241.            ADD_TNORMAL
  1242.            Tnormal->Type = DENTS;
  1243.            Tnormal->Amount = Parse_Float ();
  1244.          END_CASE
  1245.  
  1246.          CASE (RIPPLES_TOKEN)
  1247.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1248.            ADD_TNORMAL
  1249.            Tnormal->Type = RIPPLES;
  1250.            Tnormal->Amount = Parse_Float ();
  1251.          END_CASE
  1252.  
  1253.          CASE (WAVES_TOKEN)
  1254.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1255.            ADD_TNORMAL
  1256.            Tnormal->Type = WAVES;
  1257.            Tnormal->Amount = Parse_Float ();
  1258.          END_CASE
  1259.  
  1260.          CASE (WRINKLES_TOKEN)
  1261.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1262.            ADD_TNORMAL
  1263.            Tnormal->Type = WRINKLES;
  1264.            Tnormal->Amount = Parse_Float ();
  1265.          END_CASE
  1266.  
  1267.          CASE (BUMP_MAP_TOKEN)
  1268.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1269.            ADD_TNORMAL
  1270.            Parse_Bump_Map (Tnormal);
  1271.          END_CASE
  1272.  
  1273.          CASE (FREQUENCY_TOKEN)
  1274.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1275.            ADD_TNORMAL
  1276.            if (!(Tnormal->Type == RIPPLES || Tnormal->Type == WAVES))
  1277.              if (Language_Version >= 1.5)
  1278.                Warn ("Cannot use frequency with this normal",1.5);
  1279.            Tnormal->Frequency = Parse_Float();
  1280.          END_CASE
  1281.  
  1282.          CASE (PHASE_TOKEN)
  1283.            Warn_State(Token.Token_Id, TNORMAL_TOKEN);           
  1284.            ADD_TNORMAL
  1285.            if (!(Tnormal->Type == RIPPLES || Tnormal->Type == WAVES))
  1286.              if (Language_Version >= 1.5)
  1287.                Warn ("Cannot use phase with this normal",1.5);
  1288.            Tnormal->Phase = Parse_Float();
  1289.          END_CASE
  1290.  
  1291.  
  1292. /***********************************************************************
  1293. FINISH STUFF OUTSIDE FINISH{}
  1294. ***********************************************************************/
  1295.          CASE (AMBIENT_TOKEN)
  1296.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1297.            Finish->Ambient = Parse_Float ();
  1298.          END_CASE
  1299.  
  1300.          CASE (BRILLIANCE_TOKEN)
  1301.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1302.            Finish->Brilliance = Parse_Float ();
  1303.          END_CASE
  1304.  
  1305.          CASE (DIFFUSE_TOKEN)
  1306.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1307.            Finish->Diffuse = Parse_Float ();
  1308.          END_CASE
  1309.  
  1310.          CASE (REFLECTION_TOKEN)
  1311.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1312.            Finish->Reflection = Parse_Float ();
  1313.          END_CASE
  1314.  
  1315.          CASE (REFRACTION_TOKEN)
  1316.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1317.            Finish->Refraction = Parse_Float ();
  1318.          END_CASE
  1319.  
  1320.          CASE (IOR_TOKEN)
  1321.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1322.            Finish->Index_Of_Refraction = Parse_Float ();
  1323.          END_CASE
  1324.  
  1325.          CASE (PHONG_TOKEN)
  1326.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1327.            Finish->Phong = Parse_Float ();
  1328.          END_CASE
  1329.  
  1330.          CASE (PHONG_SIZE_TOKEN)
  1331.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1332.            Finish->Phong_Size = Parse_Float ();
  1333.     /*     if (Finish->Phong_Size < 1.0)
  1334.                Finish->Phong_Size = 1.0;
  1335.            if (Finish->Phong_Size > 100)
  1336.                Finish->Phong_Size = 100; */
  1337.          END_CASE
  1338.  
  1339.          CASE (SPECULAR_TOKEN)
  1340.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1341.            Finish->Specular = Parse_Float ();
  1342.          END_CASE
  1343.  
  1344.          CASE (ROUGHNESS_TOKEN)
  1345.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1346.            Finish->Roughness = Parse_Float ();
  1347.     /*     if (Finish->Roughness > 1.0)
  1348.                Finish->Roughness = 1.0;
  1349.            if (Finish->Roughness < 0.001)
  1350.                Finish->Roughness = 0.001;  */
  1351.            Finish->Roughness = 1.0/Finish->Roughness; /* CEY 12/92 */
  1352.          END_CASE
  1353.  
  1354.          CASE (METALLIC_TOKEN)
  1355.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1356.            Finish->Metallic_Flag = TRUE;
  1357.          END_CASE
  1358.  
  1359.          CASE (CRAND_TOKEN)
  1360.            Warn_State(Token.Token_Id, FINISH_TOKEN);           
  1361.            Finish->Crand = Parse_Float();
  1362.          END_CASE
  1363.  
  1364.          CASE_FLOAT
  1365.            Finish->Crand = Parse_Float();
  1366.            Warn("Should use crand keyword in finish statement.",1.5);           
  1367.          END_CASE
  1368.  
  1369.          CASE (TRANSLATE_TOKEN)
  1370.            Parse_Vector (&Local_Vector);
  1371.            Translate_Textures (Texture, &Local_Vector);
  1372.          END_CASE
  1373.  
  1374.          CASE (ROTATE_TOKEN)
  1375.            Parse_Vector (&Local_Vector);
  1376.            Rotate_Textures (Texture, &Local_Vector);
  1377.          END_CASE
  1378.  
  1379.          CASE (SCALE_TOKEN)
  1380.            Parse_Scale_Vector (&Local_Vector);
  1381.            Scale_Textures (Texture, &Local_Vector);
  1382.          END_CASE
  1383.  
  1384.          CASE (TRANSFORM_TOKEN)
  1385.            GET(TRANSFORM_ID_TOKEN)
  1386.            Transform_Textures (Texture, (TRANSFORM *)Token.Constant_Data);
  1387.          END_CASE
  1388.  
  1389.          CASE (TEXTURE_ID_TOKEN)
  1390.            Warn("Texture identifier overwriting previous values.",0);
  1391.            Destroy_Textures(Texture);
  1392.            Texture = Copy_Textures((TEXTURE *) Token.Constant_Data);
  1393.            Pigment = Texture->Pigment;
  1394.            Tnormal = Texture->Tnormal;
  1395.            Finish  = Texture->Finish;
  1396.          END_CASE
  1397.  
  1398.          OTHERWISE
  1399.            UNGET
  1400.            EXIT
  1401.          END_CASE
  1402. /***********************************************************************/
  1403.  
  1404.        END_EXPECT
  1405.  
  1406.        if (Not_In_Default && (Texture->Pigment->Type == NO_PIGMENT) &&
  1407.            !(Language_Version < 1.5))
  1408.          Parse_Error(PIGMENT_ID_TOKEN);
  1409.  
  1410.        EXIT
  1411.      END_CASE        /* End of pnf texture */
  1412.  
  1413.    END_EXPECT       /* End of texture_body */
  1414.  
  1415.    EXPECT            /* Look for texture_mods */
  1416.      CASE (TRANSLATE_TOKEN)
  1417.        Parse_Vector (&Local_Vector);
  1418.        Translate_Textures (Texture, &Local_Vector);
  1419.      END_CASE
  1420.  
  1421.      CASE (ROTATE_TOKEN)
  1422.        Parse_Vector (&Local_Vector);
  1423.        Rotate_Textures (Texture, &Local_Vector);
  1424.      END_CASE
  1425.  
  1426.      CASE (SCALE_TOKEN)
  1427.        Parse_Scale_Vector (&Local_Vector);
  1428.        Scale_Textures (Texture, &Local_Vector);
  1429.      END_CASE
  1430.  
  1431.      CASE (TRANSFORM_TOKEN)
  1432.        GET(TRANSFORM_ID_TOKEN)
  1433.        Transform_Textures (Texture, (TRANSFORM *)Token.Constant_Data);
  1434.      END_CASE
  1435.  
  1436.      OTHERWISE
  1437.        UNGET
  1438.        EXIT
  1439.      END_CASE
  1440.    END_EXPECT        /* End of texture */
  1441.  
  1442.    Parse_End ();
  1443.    return (Texture);
  1444.   }
  1445.  
  1446. static
  1447. OBJECT *Parse_Bound_Clip ()
  1448.   {
  1449.    VECTOR Local_Vector;
  1450.    OBJECT *First, *Current, *Prev;
  1451.  
  1452.    First = Prev = NULL;
  1453.  
  1454.    while ((Current = Parse_Object ()) != NULL)
  1455.      {
  1456.       if (Current->Type & (TEXTURED_OBJECT+PATCH_OBJECT))
  1457.         Error ("Illegal texture or patch in clip or bound");
  1458.       if (First == NULL)
  1459.         First = Current;
  1460.       if (Prev != NULL)
  1461.         Prev->Sibling = Current;
  1462.       Prev = Current;
  1463.      }
  1464.  
  1465.    EXPECT
  1466.      CASE (TRANSLATE_TOKEN)
  1467.        Parse_Vector (&Local_Vector);
  1468.        for (Current = First;
  1469.             Current != NULL;
  1470.             Current = Current->Sibling)
  1471.          Translate_Object (Current, &Local_Vector);
  1472.      END_CASE
  1473.  
  1474.      CASE (ROTATE_TOKEN)
  1475.        Parse_Vector (&Local_Vector);
  1476.        for (Current = First;
  1477.             Current != NULL;
  1478.             Current = Current->Sibling)
  1479.          Rotate_Object (Current, &Local_Vector);
  1480.      END_CASE
  1481.  
  1482.      CASE (SCALE_TOKEN)
  1483.        Parse_Scale_Vector (&Local_Vector);
  1484.        for (Current = First;
  1485.             Current != NULL;
  1486.             Current = Current->Sibling)
  1487.          Scale_Object (Current, &Local_Vector);
  1488.      END_CASE
  1489.  
  1490.      CASE (TRANSFORM_TOKEN)
  1491.        GET(TRANSFORM_ID_TOKEN)
  1492.        for (Current = First;
  1493.             Current != NULL;
  1494.             Current = Current->Sibling)
  1495.        Transform_Object (Current, (TRANSFORM *)Token.Constant_Data);
  1496.      END_CASE
  1497.  
  1498.      OTHERWISE
  1499.        UNGET
  1500.        EXIT
  1501.      END_CASE
  1502.    END_EXPECT
  1503.  
  1504.    return (First);
  1505.   }
  1506.  
  1507. static void Parse_Object_Mods (Object)
  1508.   OBJECT *Object;
  1509.   {
  1510.    VECTOR Local_Vector;
  1511.    TEXTURE *Local_Texture;
  1512.    COLOUR Local_Colour;
  1513.  
  1514.    EXPECT
  1515.      CASE (COLOUR_TOKEN)
  1516.        Parse_Colour (&Local_Colour);
  1517.        if (Language_Version < 1.5)
  1518.          if (Object->Texture != NULL) 
  1519.            if (Object->Texture->Type == PNF_TEXTURE)
  1520.              {
  1521.               Object->Texture->Pigment->Quick_Colour = Local_Colour;
  1522.               break;  /* acts like END_CASE */
  1523.              }
  1524.        Warn("Quick color belongs in texture.  Color ignored.",0.0);
  1525.      END_CASE
  1526.  
  1527.      CASE (TRANSLATE_TOKEN)
  1528.        Parse_Vector (&Local_Vector);
  1529.        Translate_Object (Object, &Local_Vector);
  1530.      END_CASE
  1531.  
  1532.      CASE (ROTATE_TOKEN)
  1533.        Parse_Vector (&Local_Vector);
  1534.        Rotate_Object (Object, &Local_Vector);
  1535.      END_CASE
  1536.  
  1537.      CASE (SCALE_TOKEN)
  1538.        Parse_Scale_Vector (&Local_Vector);
  1539.        Scale_Object (Object, &Local_Vector);
  1540.      END_CASE
  1541.  
  1542.      CASE (TRANSFORM_TOKEN)
  1543.        GET(TRANSFORM_ID_TOKEN)
  1544.        Transform_Object (Object, (TRANSFORM *)Token.Constant_Data);
  1545.      END_CASE
  1546.  
  1547.      CASE (BOUNDED_BY_TOKEN)
  1548.        Parse_Begin ();
  1549.        if (Object->Bound != NULL)
  1550.          Error ("Cannot have more than one BOUNDED_BY {} per object");
  1551.  
  1552.        EXPECT
  1553.          CASE (CLIPPED_BY_TOKEN)
  1554.            Object->Bound = Object->Clip;
  1555.            EXIT
  1556.          END_CASE
  1557.  
  1558.          OTHERWISE
  1559.            UNGET
  1560.            Object->Bound = Parse_Bound_Clip ();
  1561.            EXIT
  1562.          END_CASE
  1563.        END_EXPECT
  1564.        
  1565.        Parse_End ();
  1566.      END_CASE
  1567.  
  1568.      CASE (CLIPPED_BY_TOKEN)
  1569.        Parse_Begin ();
  1570.        if (Object->Clip != NULL)
  1571.          Error ("Cannot have more than one CLIPPED_BY {} per object");
  1572.  
  1573.        EXPECT
  1574.          CASE (BOUNDED_BY_TOKEN)
  1575.            Object->Clip = Object->Bound;
  1576.            EXIT
  1577.          END_CASE
  1578.  
  1579.          OTHERWISE
  1580.            UNGET
  1581.            Object->Clip = Parse_Bound_Clip ();
  1582.            EXIT
  1583.          END_CASE
  1584.        END_EXPECT
  1585.  
  1586.        Parse_End ();
  1587.      END_CASE
  1588.  
  1589.      CASE (TEXTURE_TOKEN)
  1590.        Object->Type |= TEXTURED_OBJECT;
  1591.        Local_Texture = Parse_Texture ();
  1592.        Link_Textures(&(Object->Texture), Local_Texture);
  1593.      END_CASE
  1594.  
  1595.      CASE3 (PIGMENT_TOKEN, TNORMAL_TOKEN, FINISH_TOKEN)
  1596.        Object->Type |= TEXTURED_OBJECT;
  1597.        if (Object->Texture == NULL)
  1598.          Object->Texture = Copy_Textures(Default_Texture);
  1599.        else
  1600.          if (Object->Texture->Type != PNF_TEXTURE)
  1601.            Link_Textures(&(Object->Texture), Copy_Textures(Default_Texture));
  1602.        UNGET
  1603.        EXPECT
  1604.          CASE (PIGMENT_TOKEN)
  1605.            Parse_Pigment ( &(Object->Texture->Pigment) );
  1606.          END_CASE
  1607.  
  1608.          CASE (TNORMAL_TOKEN)
  1609.            Parse_Tnormal ( &(Object->Texture->Tnormal) );
  1610.          END_CASE
  1611.  
  1612.          CASE (FINISH_TOKEN)
  1613.            Parse_Finish ( &(Object->Texture->Finish) );
  1614.          END_CASE
  1615.  
  1616.          OTHERWISE
  1617.            UNGET
  1618.            EXIT
  1619.          END_CASE
  1620.        END_EXPECT
  1621.      END_CASE
  1622.  
  1623.      CASE (INVERSE_TOKEN)
  1624.        if (Object->Type & PATCH_OBJECT)
  1625.          Warn ("Cannot invert a patch object",0.0);
  1626.        Invert_Object (Object);
  1627.      END_CASE
  1628.  
  1629.      CASE (STURM_TOKEN)
  1630.        if (!(Object->Type & STURM_OK_OBJECT))
  1631.          Error ("Cannot use STRUM here");
  1632.        ((POLY *) Object)->Sturm_Flag = TRUE;
  1633.      END_CASE
  1634.  
  1635.      CASE (WATER_LEVEL_TOKEN)
  1636.        if (!(Object->Type & WATER_LEVEL_OK_OBJECT))
  1637.          Error ("Cannot use WATER_LEVEL here");
  1638.        ((HEIGHT_FIELD *) Object)->bounding_box->bounds[0].y = 65536.0 * Parse_Float();
  1639.      END_CASE
  1640.  
  1641.      CASE (SMOOTH_TOKEN)
  1642.        if (!(Object->Type & SMOOTH_OK_OBJECT))
  1643.          Error ("Cannot use SMOOTH here");
  1644.        ((HEIGHT_FIELD *) Object)->Smoothed = TRUE;
  1645.      END_CASE
  1646.  
  1647.      CASE (NO_SHADOW_TOKEN)
  1648.        Object->No_Shadow_Flag = TRUE;
  1649.      END_CASE
  1650.  
  1651.      CASE (LIGHT_SOURCE_TOKEN)
  1652.        Error("Light source must be defined using new syntax");
  1653.      END_CASE
  1654.  
  1655.      OTHERWISE
  1656.        UNGET
  1657.        EXIT
  1658.      END_CASE
  1659.    END_EXPECT
  1660.  
  1661.    if (Object->Bound != NULL)
  1662.      {
  1663.       Object->Bounds.Lower_Left = Object->Bound->Bounds.Lower_Left;
  1664.       Object->Bounds.Lengths    = Object->Bound->Bounds.Lengths;
  1665.      }
  1666.    Parse_End ();
  1667.   }
  1668.  
  1669. static
  1670. OBJECT *Parse_Sphere ()
  1671.   {
  1672.    SPHERE *Object;
  1673.  
  1674.    Parse_Begin ();
  1675.  
  1676.    if ( (Object = (SPHERE *)Parse_Object_Id()) != NULL)
  1677.       return ((OBJECT *) Object);
  1678.       
  1679.    Object = Create_Sphere();
  1680.  
  1681.    Parse_Vector(&(Object -> Center));   Parse_Comma();
  1682.    Object -> Radius = Parse_Float();
  1683.    Object -> Radius_Squared = Object -> Radius * Object -> Radius;
  1684.    Object -> Inverse_Radius = 1.0 / Object -> Radius;
  1685.  
  1686.    Make_Vector(&Object->Bounds.Lower_Left,
  1687.            Object->Center.x - Object->Radius,
  1688.            Object->Center.y - Object->Radius,
  1689.            Object->Center.z - Object->Radius);
  1690.    Make_Vector(&Object->Bounds.Lengths,
  1691.            2.0 * Object->Radius,
  1692.            2.0 * Object->Radius,
  1693.            2.0 * Object->Radius);
  1694.  
  1695.    Parse_Object_Mods ((OBJECT *) Object);
  1696.  
  1697.    return ((OBJECT *) Object);
  1698.   }
  1699.  
  1700. static
  1701. OBJECT *Parse_Plane ()
  1702.   {
  1703.    PLANE *Object;
  1704.  
  1705.    Parse_Begin ();
  1706.  
  1707.    if ( (Object = (PLANE *)Parse_Object_Id()) != NULL)
  1708.       return ((OBJECT *) Object);
  1709.       
  1710.    Object = Create_Plane();
  1711.  
  1712.    Parse_Vector(&(Object -> Normal_Vector));   Parse_Comma();
  1713.    VNormalize(Object->Normal_Vector, Object->Normal_Vector);
  1714.    Object->Distance = -Parse_Float();
  1715.  
  1716.    Parse_Object_Mods ((OBJECT *)Object);
  1717.  
  1718.    return ((OBJECT *) Object);
  1719.   }
  1720.  
  1721. static
  1722. OBJECT *Parse_Height_Field ()
  1723.   {
  1724.    HEIGHT_FIELD *Object;
  1725.    VECTOR Local_Vector;
  1726.    IMAGE *Image;
  1727.  
  1728.    Parse_Begin ();
  1729.  
  1730.    if ( (Object = (HEIGHT_FIELD *)Parse_Object_Id()) != NULL)
  1731.       return ((OBJECT *) Object);
  1732.       
  1733.    Object = Create_Height_Field();
  1734.  
  1735.    Image = Parse_Image (HF_FILE);
  1736.    Image->Use_Colour_Flag = FALSE;
  1737.  
  1738.    Object->bounding_box->bounds[0].x = 1.0;
  1739.    Object->bounding_box->bounds[0].y = 0.0;
  1740.    Object->bounding_box->bounds[0].z = 1.0;
  1741.    if (Image->File_Type == POT_FILE)
  1742.      {
  1743.       Object->bounding_box->bounds[1].x = Image -> width/2.0 - 2.0;
  1744.       Make_Vector(&Local_Vector,2.0/Image->width,1.0/65536.0,1.0/Image->height);
  1745.      }
  1746.    else
  1747.      {
  1748.       Object->bounding_box->bounds[1].x = Image -> width - 2.0;
  1749.       Make_Vector(&Local_Vector,1.0/(Image->width),1.0/65536.0,1.0/(Image->height));
  1750.      }
  1751.    Object->bounding_box->bounds[1].y = 65536.0;
  1752.    Object->bounding_box->bounds[1].z = Image -> height - 2.0;
  1753.    Compute_Scaling_Transform(Object->Trans,&Local_Vector);
  1754.  
  1755.    Parse_Object_Mods ((OBJECT *)Object);
  1756.  
  1757.    Find_Hf_Min_Max(Object, Image);
  1758.  
  1759.    return ((OBJECT *) Object);
  1760.   }
  1761.  
  1762. static
  1763. OBJECT *Parse_Triangle ()
  1764.   {
  1765.    TRIANGLE *Object;
  1766.  
  1767.    Parse_Begin ();
  1768.  
  1769.    if ( (Object = (TRIANGLE *)Parse_Object_Id()) != NULL)
  1770.       return ((OBJECT *) Object);
  1771.       
  1772.    Object = Create_Triangle();
  1773.  
  1774.    Parse_Vector (&Object->P1);    Parse_Comma();
  1775.    Parse_Vector (&Object->P2);    Parse_Comma();
  1776.    Parse_Vector (&Object->P3);
  1777.    if (!Compute_Triangle (Object,FALSE))
  1778.      fprintf (stdout, "Degenerate triangle on line %d.  Please remove.\n",
  1779.               Token.Token_Line_No+1);
  1780.  
  1781.    Parse_Object_Mods ((OBJECT *)Object);
  1782.  
  1783.    return ((OBJECT *) Object);
  1784.   }
  1785.  
  1786. static
  1787. OBJECT *Parse_Smooth_Triangle ()
  1788.   {
  1789.    SMOOTH_TRIANGLE *Object;
  1790.    short degen;                                                   /* LSK */
  1791.    DBL vlen;                                                      /* LSK */
  1792.  
  1793.    degen=FALSE;
  1794.  
  1795.    Parse_Begin ();
  1796.  
  1797.    if ( (Object = (SMOOTH_TRIANGLE *)Parse_Object_Id()) != NULL)
  1798.       return ((OBJECT *) Object);
  1799.  
  1800.    Object = Create_Smooth_Triangle();
  1801.  
  1802.    Parse_Vector (&Object->P1);    Parse_Comma();
  1803.    Parse_Vector (&Object->N1);    Parse_Comma();
  1804.  
  1805.    VLength(vlen,Object->N1);                                     /* LSK */
  1806.    if (vlen<1E-09)                                               /* LSK */
  1807.      degen=TRUE;                                                 /* LSK */
  1808.    else                                                          /* LSK */
  1809.      VNormalize (Object->N1, Object->N1);
  1810.  
  1811.    Parse_Vector (&Object->P2);    Parse_Comma();
  1812.    Parse_Vector (&Object->N2);    Parse_Comma();
  1813.  
  1814.    VLength(vlen,Object->N2);                                     /* LSK */
  1815.    if (vlen<1E-09)                                               /* LSK */
  1816.      degen=TRUE;                                                 /* LSK */
  1817.    else                                                          /* LSK */
  1818.      VNormalize (Object->N2, Object->N2);
  1819.  
  1820.    Parse_Vector (&Object->P3);    Parse_Comma();
  1821.    Parse_Vector (&Object->N3);
  1822.  
  1823.    VLength(vlen,Object->N3);                                     /* LSK */
  1824.    if (vlen<1E-09)                                               /* LSK */
  1825.      degen=TRUE;                                                 /* LSK */
  1826.    else                                                          /* LSK */
  1827.      VNormalize (Object->N3, Object->N3);
  1828.  
  1829.    if (!degen) {                                                 /* LSK */
  1830.      degen=!Compute_Triangle ((TRIANGLE *) Object,TRUE);         /* LSK */
  1831.    }
  1832.  
  1833.    if (degen)                                                    /* LSK */
  1834.      fprintf (stdout, "Degenerate triangle on line %d.  Please remove.\n",
  1835.               Token.Token_Line_No+1);
  1836.  
  1837.    Parse_Object_Mods ((OBJECT *)Object);
  1838.  
  1839.    return ((OBJECT *) Object);
  1840.  }
  1841.  
  1842. static
  1843. OBJECT *Parse_Quadric ()
  1844.   {
  1845.    QUADRIC *Object;
  1846.  
  1847.    Parse_Begin ();
  1848.  
  1849.    if ( (Object = (QUADRIC *)Parse_Object_Id()) != NULL)
  1850.       return ((OBJECT *) Object);
  1851.       
  1852.    Object = Create_Quadric();
  1853.  
  1854.    Parse_Vector(&(Object -> Square_Terms));     Parse_Comma();
  1855.    Parse_Vector(&(Object -> Mixed_Terms));      Parse_Comma();
  1856.    Parse_Vector(&(Object -> Terms));            Parse_Comma();
  1857.    Object -> Constant = Parse_Float();
  1858.    Object -> Non_Zero_Square_Term =
  1859.      !( (Object -> Square_Terms.x == 0.0)
  1860.      && (Object -> Square_Terms.y == 0.0)
  1861.      && (Object -> Square_Terms.z == 0.0)
  1862.      && (Object -> Mixed_Terms.x == 0.0)
  1863.      && (Object -> Mixed_Terms.y == 0.0)
  1864.      && (Object -> Mixed_Terms.z == 0.0));
  1865.  
  1866.    Parse_Object_Mods ((OBJECT *)Object);
  1867.  
  1868.    return ((OBJECT *) Object);
  1869.   }
  1870.  
  1871. static
  1872. OBJECT *Parse_Box ()
  1873.   {
  1874.    BOX *Object;
  1875.  
  1876.    Parse_Begin ();
  1877.  
  1878.    if ( (Object = (BOX *)Parse_Object_Id()) != NULL)
  1879.       return ((OBJECT *) Object);
  1880.       
  1881.    Object = Create_Box();
  1882.  
  1883.    Parse_Vector(&(Object->bounds[0]));     Parse_Comma();
  1884.    Parse_Vector(&(Object->bounds[1]));
  1885.    
  1886.    Object->Bounds.Lower_Left=Object->bounds[0];
  1887.    VSub(Object->Bounds.Lengths, Object->bounds[1],Object->bounds[0]);
  1888.  
  1889.    Parse_Object_Mods ((OBJECT *)Object);
  1890.  
  1891.    return ((OBJECT *) Object);
  1892.   }
  1893.  
  1894. static
  1895. OBJECT *Parse_Disc ()
  1896.   {
  1897.    DISC *Object;
  1898.    DBL tmpf;
  1899.    VECTOR lengths;
  1900.  
  1901.    Parse_Begin ();
  1902.  
  1903.    if ( (Object = (DISC *)Parse_Object_Id()) != NULL)
  1904.       return ((OBJECT *) Object);
  1905.       
  1906.    Object = Create_Disc();
  1907.  
  1908.    Parse_Vector(&(Object->center)); Parse_Comma ();
  1909.    Parse_Vector(&(Object->normal)); Parse_Comma ();
  1910.    VNormalize(Object->normal, Object->normal);
  1911.  
  1912.    tmpf = Parse_Float(); Parse_Comma ();
  1913.    Object->oradius2 = tmpf * tmpf;
  1914.  
  1915.    EXPECT
  1916.      CASE_FLOAT
  1917.        tmpf = Parse_Float();
  1918.        Object->iradius2 = tmpf * tmpf;
  1919.      END_CASE
  1920.  
  1921.      OTHERWISE
  1922.        UNGET
  1923.        EXIT
  1924.      END_CASE
  1925.    END_EXPECT
  1926.  
  1927.    /* Calculate info needed for ray-disc intersections */
  1928.    VDot(tmpf, Object->center, Object->normal);
  1929.    Object->d = -tmpf;
  1930.  
  1931.    /* Calculate the bounds */
  1932.    tmpf = sqrt(Object->oradius2);
  1933.    Make_Vector(&lengths, tmpf, tmpf, tmpf);
  1934.    VSub(Object->Bounds.Lower_Left, Object->center, lengths);
  1935.    VScale(Object->Bounds.Lengths, lengths, 2.0);
  1936.  
  1937.    Parse_Object_Mods ((OBJECT *)Object);
  1938.  
  1939.    return ((OBJECT *) Object);
  1940.   }
  1941.  
  1942. static
  1943. OBJECT *Parse_Cylinder ()
  1944.   {
  1945.    CONE *Object;
  1946.  
  1947.    Parse_Begin ();
  1948.  
  1949.    if ( (Object = (CONE *)Parse_Object_Id()) != NULL)
  1950.       return ((OBJECT *) Object);
  1951.       
  1952.    Object = Create_Cylinder();
  1953.  
  1954.    Parse_Vector(&(Object->apex));  Parse_Comma ();
  1955.    Parse_Vector(&(Object->base));  Parse_Comma ();
  1956.    Object->apex_radius = Parse_Float();
  1957.    Object->base_radius = Object->apex_radius;
  1958.  
  1959.    EXPECT
  1960.      CASE(OPEN_TOKEN)
  1961.        Object->closed = 0;
  1962.        EXIT
  1963.      END_CASE
  1964.      
  1965.      OTHERWISE
  1966.        UNGET
  1967.        EXIT
  1968.      END_CASE
  1969.    END_EXPECT
  1970.  
  1971.    Compute_Cylinder_Data((OBJECT *)Object);
  1972.  
  1973.    Parse_Object_Mods ((OBJECT *)Object);
  1974.  
  1975.    return ((OBJECT *) Object);
  1976.   }
  1977.  
  1978. static
  1979. OBJECT *Parse_Cone ()
  1980.   {
  1981.    CONE *Object;
  1982.  
  1983.    Parse_Begin ();
  1984.  
  1985.    if ( (Object = (CONE *)Parse_Object_Id()) != NULL)
  1986.       return ((OBJECT *) Object);
  1987.       
  1988.    Object = Create_Cone();
  1989.  
  1990.    Parse_Vector(&(Object->apex));  Parse_Comma ();
  1991.    Object->apex_radius = Parse_Float();  Parse_Comma ();
  1992.  
  1993.    Parse_Vector(&(Object->base));  Parse_Comma ();
  1994.    Object->base_radius = Parse_Float();
  1995.    
  1996.    EXPECT
  1997.      CASE(OPEN_TOKEN)
  1998.        Object->closed = 0;
  1999.        EXIT
  2000.      END_CASE
  2001.      
  2002.      OTHERWISE
  2003.        UNGET
  2004.        EXIT
  2005.      END_CASE
  2006.    END_EXPECT
  2007.  
  2008.    /* Compute run-time values for the cone */
  2009.    Compute_Cone_Data((OBJECT *)Object);
  2010.  
  2011.    Parse_Object_Mods ((OBJECT *)Object);
  2012.  
  2013.    return ((OBJECT *) Object);
  2014.   }
  2015.  
  2016. static
  2017. OBJECT *Parse_Blob ()
  2018.   {
  2019.    BLOB *Object;
  2020.    DBL threshold;
  2021.    int npoints;
  2022.    blobstackptr blob_components, blob_component;
  2023.  
  2024.    Parse_Begin ();
  2025.  
  2026.    if ( (Object = (BLOB *)Parse_Object_Id()) != NULL)
  2027.       return ((OBJECT *) Object);
  2028.       
  2029.    Object = Create_Blob();
  2030.  
  2031.    blob_components = NULL;
  2032.    npoints = 0;
  2033.    threshold = 1.0;
  2034.  
  2035.    EXPECT
  2036.      CASE (THRESHOLD_TOKEN)
  2037.        threshold = Parse_Float();
  2038.      END_CASE
  2039.  
  2040.      CASE (COMPONENT_TOKEN)
  2041.        blob_component = (blobstackptr) malloc(sizeof(struct blob_list_struct));
  2042.        if (blob_component == NULL)
  2043.           MAError("blob component");
  2044.        blob_component->elem.coeffs[2] = Parse_Float(); Parse_Comma();
  2045.        blob_component->elem.radius2   = Parse_Float(); Parse_Comma();
  2046.        Parse_Vector(&blob_component->elem.pos);
  2047.        blob_component->next = blob_components;
  2048.        blob_components = blob_component;
  2049.        npoints++;
  2050.      END_CASE
  2051.  
  2052.      OTHERWISE
  2053.        UNGET
  2054.        EXIT
  2055.      END_CASE
  2056.    END_EXPECT
  2057.  
  2058.    /* Finally, process the information */
  2059.    MakeBlob(Object, threshold, blob_components, npoints, 0);
  2060.  
  2061.    Parse_Object_Mods ((OBJECT *)Object);
  2062.  
  2063.    return ((OBJECT *) Object);
  2064.   }
  2065.  
  2066. static
  2067. OBJECT *Parse_Torus ()
  2068.   {
  2069.    POLY *Object;
  2070.    DBL iradius, oradius, *Coeffs;
  2071.  
  2072.    Parse_Begin ();
  2073.  
  2074.    if ( (Object = (POLY *)Parse_Object_Id()) != NULL)
  2075.       return ((OBJECT *) Object);
  2076.       
  2077.    Object = Create_Poly(4);
  2078.  
  2079.    /* Read in the two radii */
  2080.    iradius = Parse_Float(); /* Big radius */
  2081.    Parse_Comma();
  2082.    oradius = Parse_Float(); /* Little radius */
  2083.  
  2084.    /* Build the coefficients of a torus lying in the x-z plane */
  2085.    Coeffs = Object->Coeffs;
  2086.    Coeffs[ 0] =  1.0;
  2087.    Coeffs[ 4] =  2.0;
  2088.    Coeffs[ 7] =  2.0;
  2089.    Coeffs[ 9] = -2.0 * (iradius * iradius + oradius * oradius);
  2090.    Coeffs[20] =  1.0;
  2091.    Coeffs[23] =  2.0;
  2092.    Coeffs[25] =  2.0 * (iradius * iradius - oradius * oradius);
  2093.    Coeffs[30] =  1.0;
  2094.    Coeffs[32] = -2.0 * (iradius * iradius + oradius * oradius);
  2095.    Coeffs[34] = (iradius * iradius - oradius * oradius) *
  2096.             (iradius * iradius - oradius * oradius);
  2097.  
  2098.    Make_Vector(&Object->Bounds.Lower_Left, -(iradius + oradius),
  2099.            -iradius, -(iradius + oradius))
  2100.    Make_Vector(&Object->Bounds.Lengths, 2.0 * (iradius + oradius),
  2101.            2.0 * iradius, 2.0 * (iradius + oradius));
  2102.  
  2103.    Parse_Object_Mods ((OBJECT *)Object);
  2104.  
  2105.    return ((OBJECT *) Object);
  2106.   }
  2107.  
  2108. static
  2109. OBJECT *Parse_Poly (order)
  2110.   int order;
  2111.   {
  2112.    POLY *Object;
  2113.  
  2114.    Parse_Begin ();
  2115.  
  2116.    if ( (Object = (POLY *)Parse_Object_Id()) != NULL)
  2117.       return ((OBJECT *) Object);
  2118.       
  2119.    if (order == 0)
  2120.      {
  2121.       order = (int)Parse_Float();      Parse_Comma();
  2122.       if (order < 2 || order > MAX_ORDER)
  2123.         Error("Order of poly is out of range");
  2124.      }
  2125.  
  2126.    Object = Create_Poly(order);
  2127.  
  2128.    Parse_Coeffs(Object->Order, &(Object->Coeffs[0]));
  2129.  
  2130.    Parse_Object_Mods ((OBJECT *)Object);
  2131.  
  2132.    return ((OBJECT *) Object);
  2133.   }
  2134.  
  2135. static
  2136. OBJECT *Parse_Bicubic_Patch ()
  2137.   {
  2138.    BICUBIC_PATCH *Object;
  2139.    int i, j;
  2140.  
  2141.    Parse_Begin ();
  2142.  
  2143.    if ( (Object = (BICUBIC_PATCH *)Parse_Object_Id()) != NULL)
  2144.       return ((OBJECT *) Object);
  2145.       
  2146.    Object = Create_Bicubic_Patch();
  2147.  
  2148.    EXPECT
  2149.      CASE_FLOAT
  2150.        Warn("Should use keywords for bicubic parameters.",1.5);
  2151.        Object->Patch_Type = (int)Parse_Float();
  2152.        if (Object->Patch_Type == 2 ||
  2153.            Object->Patch_Type == 3)
  2154.            Object->Flatness_Value = Parse_Float();
  2155.          else
  2156.            Object->Flatness_Value = 0.1;
  2157.        Object->U_Steps = (int)Parse_Float();
  2158.        Object->V_Steps = (int)Parse_Float();
  2159.        EXIT
  2160.      END_CASE       
  2161.        
  2162.      CASE (TYPE_TOKEN)
  2163.        Object->Patch_Type = (int)Parse_Float();
  2164.      END_CASE
  2165.  
  2166.      CASE (FLATNESS_TOKEN)
  2167.        Object->Flatness_Value = Parse_Float();
  2168.      END_CASE
  2169.  
  2170.      CASE (V_STEPS_TOKEN)
  2171.        Object->V_Steps = (int)Parse_Float();
  2172.      END_CASE
  2173.  
  2174.      CASE (U_STEPS_TOKEN)
  2175.        Object->U_Steps = (int)Parse_Float();
  2176.      END_CASE
  2177.  
  2178.      OTHERWISE
  2179.        UNGET
  2180.        EXIT
  2181.      END_CASE
  2182.    END_EXPECT
  2183.  
  2184.    if (Object->Patch_Type > 1)
  2185.      {
  2186.       Object->Patch_Type = 1;
  2187.       Warn("Patch type no longer supported. Using type 1.",0.0);
  2188.      }
  2189.  
  2190.    if ((Object->Patch_Type < 0) || (Object->Patch_Type > MAX_PATCH_TYPE))
  2191.      Error("Undefined bicubic patch type");
  2192.  
  2193.    Parse_Comma();
  2194.    for (i=0;i<4;i++)
  2195.      for (j=0;j<4;j++)
  2196.        {
  2197.         Parse_Vector(&(Object -> Control_Points[i][j]));
  2198.         if (!((i==3)&&(j==3)))
  2199.           Parse_Comma();
  2200.        };
  2201.    Precompute_Patch_Values(Object); /* interpolated mesh coords */
  2202.  
  2203.    Parse_Object_Mods ((OBJECT *)Object);
  2204.  
  2205.    return ((OBJECT *) Object);
  2206.   }
  2207.  
  2208. static
  2209. OBJECT *Parse_CSG (CSG_Type)
  2210.   int CSG_Type;
  2211.   {
  2212.    CSG *Object;
  2213.    OBJECT *Local;
  2214.    int Object_Count = 0;
  2215.  
  2216.    Parse_Begin ();
  2217.  
  2218.    if ( (Object = (CSG *)Parse_Object_Id()) != NULL)
  2219.       return ((OBJECT *) Object);
  2220.       
  2221.    if (CSG_Type & CSG_UNION_TYPE)
  2222.      Object = Create_CSG_Union ();
  2223.    else
  2224.      if (CSG_Type & CSG_MERGE_TYPE)
  2225.        Object = Create_CSG_Merge ();
  2226.      else
  2227.        Object = Create_CSG_Intersection ();
  2228.  
  2229.    Object->Children = NULL;
  2230.  
  2231.    while ((Local = Parse_Object ()) != NULL)
  2232.      {
  2233.       if ((CSG_Type & CSG_INTERSECTION_TYPE) && (Local->Type & PATCH_OBJECT))
  2234.         Warn ("Patch objects not allowed in intersection",0.0);
  2235.       Object_Count++;
  2236.       if ((CSG_Type & CSG_DIFFERENCE_TYPE) && (Object_Count > 1))
  2237.         Invert_Object (Local);
  2238.       Object->Type |=  (Local->Type & CHILDREN_FLAGS);
  2239.       Local->Type |= IS_CHILD_OBJECT;
  2240.       Link(Local, &Local->Sibling, &Object->Children);
  2241.      };
  2242.  
  2243.    if ((Object_Count < 2) && (Language_Version >= 1.5))
  2244.      Warn ("Should have at least 2 objects in csg",1.5);
  2245.    Compute_CSG_Bounds((OBJECT *)Object);
  2246.  
  2247.    Parse_Object_Mods ((OBJECT *)Object);
  2248.  
  2249.    return ((OBJECT *) Object);
  2250.   }
  2251.  
  2252. static
  2253. OBJECT *Parse_Light_Source ()
  2254.   {
  2255.    VECTOR Local_Vector;
  2256.    LIGHT_SOURCE *Object;
  2257.  
  2258.    Parse_Begin ();
  2259.  
  2260.    if ( (Object = (LIGHT_SOURCE *)Parse_Object_Id()) != NULL)
  2261.       return ((OBJECT *) Object);
  2262.       
  2263.    Object = Create_Light_Source ();
  2264.  
  2265.    Parse_Vector(&Object->Center);
  2266.  
  2267.    GET (COLOUR_TOKEN)
  2268.  
  2269.    Parse_Colour (&Object->Colour);
  2270.  
  2271.    EXPECT
  2272.      CASE (LOOKS_LIKE_TOKEN)
  2273.        if (Object->Children != NULL)
  2274.          Error("Only one looks_like allowed per light_source");
  2275.        Parse_Begin ();
  2276.        Object->Type &= ~(int)PATCH_OBJECT;
  2277.        if ((Object->Children = Parse_Object ()) == NULL)
  2278.          Parse_Error_Str ("object");
  2279.        Translate_Object (Object->Children, &Object->Center);
  2280.        Parse_Object_Mods (Object->Children);
  2281.        Object->Children->No_Shadow_Flag = TRUE;
  2282.        Object->No_Shadow_Flag = TRUE;
  2283.        Object->Type |= (Object->Children->Type & CHILDREN_FLAGS);
  2284.      END_CASE
  2285.  
  2286.      CASE (SPOTLIGHT_TOKEN)
  2287.        Object->Light_Type = SPOT_SOURCE;
  2288.      END_CASE
  2289.  
  2290.      CASE (POINT_AT_TOKEN)
  2291.        if (Object->Light_Type == SPOT_SOURCE)
  2292.          Parse_Vector(&Object->Points_At);
  2293.        else
  2294.          Error("Spotlight param illegal in standard light source");
  2295.      END_CASE
  2296.  
  2297.      CASE (TIGHTNESS_TOKEN)
  2298.        if (Object->Light_Type == SPOT_SOURCE)
  2299.          Object->Coeff = Parse_Float();
  2300.        else
  2301.          Error("Spotlight param illegal in standard light source");
  2302.      END_CASE
  2303.  
  2304.      CASE (RADIUS_TOKEN)
  2305.        if (Object->Light_Type == SPOT_SOURCE)
  2306.          Object->Radius = cos(Parse_Float() * M_PI / 180.0);
  2307.        else
  2308.          Error("Spotlight param illegal in standard light source");
  2309.      END_CASE
  2310.  
  2311.      CASE (FALLOFF_TOKEN)
  2312.        if (Object->Light_Type == SPOT_SOURCE)
  2313.          Object->Falloff = cos(Parse_Float() * M_PI / 180.0);
  2314.        else
  2315.          Error("Spotlight param illegal in standard light source");
  2316.      END_CASE
  2317.  
  2318.      CASE (AREA_LIGHT_TOKEN)
  2319.        Object -> Area_Light = TRUE;
  2320.        Parse_Vector (&(Object -> Axis1)); Parse_Comma ();
  2321.        Parse_Vector (&(Object -> Axis2)); Parse_Comma ();
  2322.        Object -> Area_Size1 = (int)Parse_Float(); Parse_Comma ();
  2323.        Object -> Area_Size2 = (int)Parse_Float();
  2324.        Object -> Light_Grid = Create_Light_Grid (Object -> Area_Size1,
  2325.             Object -> Area_Size2);
  2326.      END_CASE
  2327.  
  2328.      CASE (JITTER_TOKEN)
  2329.        Object -> Jitter = TRUE;
  2330.      END_CASE
  2331.  
  2332.      CASE (TRACK_TOKEN)
  2333.        Object -> Track = TRUE;
  2334.      END_CASE
  2335.  
  2336.      CASE (ADAPTIVE_TOKEN)
  2337.        Object -> Adaptive_Level = (int)Parse_Float();
  2338.      END_CASE
  2339.  
  2340.      CASE (TRANSLATE_TOKEN)
  2341.        Parse_Vector (&Local_Vector);
  2342.        Translate_Object ((OBJECT *)Object, &Local_Vector);
  2343.      END_CASE
  2344.  
  2345.      CASE (ROTATE_TOKEN)
  2346.        Parse_Vector (&Local_Vector);
  2347.        Rotate_Object ((OBJECT *)Object, &Local_Vector);
  2348.      END_CASE
  2349.  
  2350.      CASE (SCALE_TOKEN)
  2351.        Parse_Scale_Vector (&Local_Vector);
  2352.        Scale_Object ((OBJECT *)Object, &Local_Vector);
  2353.      END_CASE
  2354.  
  2355.      CASE (TRANSFORM_TOKEN)
  2356.        GET(TRANSFORM_ID_TOKEN)
  2357.        Transform_Object ((OBJECT *)Object, (TRANSFORM *)Token.Constant_Data);
  2358.      END_CASE
  2359.  
  2360.      OTHERWISE
  2361.        UNGET
  2362.        EXIT
  2363.      END_CASE
  2364.    END_EXPECT
  2365.  
  2366.    Parse_End ();
  2367.  
  2368.    return ((OBJECT *)Object);
  2369.   }
  2370.  
  2371.  
  2372. static
  2373. OBJECT *Parse_Object ()
  2374.   {
  2375.    OBJECT *Object = NULL;
  2376.  
  2377.    EXPECT
  2378.      CASE (SPHERE_TOKEN)
  2379.        Object = Parse_Sphere ();
  2380.        EXIT
  2381.      END_CASE
  2382.  
  2383.      CASE (PLANE_TOKEN)
  2384.        Object = Parse_Plane ();
  2385.        EXIT
  2386.      END_CASE
  2387.  
  2388.      CASE (CONE_TOKEN)
  2389.        Object = Parse_Cone ();
  2390.        EXIT
  2391.      END_CASE
  2392.  
  2393.      CASE (CYLINDER_TOKEN)
  2394.        Object = Parse_Cylinder ();
  2395.        EXIT
  2396.      END_CASE
  2397.  
  2398.      CASE (DISC_TOKEN)
  2399.        Object = Parse_Disc ();
  2400.        EXIT
  2401.      END_CASE
  2402.  
  2403.      CASE (QUADRIC_TOKEN)
  2404.        Object = Parse_Quadric ();
  2405.        EXIT
  2406.      END_CASE
  2407.  
  2408.      CASE (CUBIC_TOKEN)
  2409.        Object = Parse_Poly (3);
  2410.        EXIT
  2411.      END_CASE
  2412.  
  2413.      CASE (QUARTIC_TOKEN)
  2414.        Object = Parse_Poly (4);
  2415.        EXIT
  2416.      END_CASE
  2417.  
  2418.      CASE (POLY_TOKEN)
  2419.        Object = Parse_Poly (0);
  2420.        EXIT
  2421.      END_CASE
  2422.  
  2423.      CASE (TORUS_TOKEN)
  2424.        Object = Parse_Torus ();
  2425.        EXIT
  2426.      END_CASE
  2427.  
  2428.      CASE (OBJECT_ID_TOKEN)
  2429.        Object = Copy_Object((OBJECT *) Token.Constant_Data);
  2430.        EXIT
  2431.      END_CASE
  2432.  
  2433.      CASE (UNION_TOKEN)
  2434.        Object = Parse_CSG (CSG_UNION_TYPE);
  2435.        EXIT
  2436.      END_CASE
  2437.  
  2438.      CASE (COMPOSITE_TOKEN)
  2439.        Warn("Use union instead of composite",1.5);
  2440.        Object = Parse_CSG (CSG_UNION_TYPE);
  2441.        EXIT
  2442.      END_CASE
  2443.  
  2444.      CASE (MERGE_TOKEN)
  2445.        Object = Parse_CSG (CSG_MERGE_TYPE);
  2446.        EXIT
  2447.      END_CASE
  2448.  
  2449.      CASE (INTERSECTION_TOKEN)
  2450.        Object = Parse_CSG (CSG_INTERSECTION_TYPE);
  2451.        EXIT
  2452.      END_CASE
  2453.  
  2454.      CASE (DIFFERENCE_TOKEN)
  2455.        Object = Parse_CSG (CSG_DIFFERENCE_TYPE+CSG_INTERSECTION_TYPE);
  2456.        EXIT
  2457.      END_CASE
  2458.  
  2459.      CASE (BICUBIC_PATCH_TOKEN)
  2460.        Object = Parse_Bicubic_Patch ();
  2461.        EXIT
  2462.      END_CASE
  2463.  
  2464.      CASE (TRIANGLE_TOKEN)
  2465.        Object = Parse_Triangle ();
  2466.        EXIT
  2467.      END_CASE
  2468.  
  2469.      CASE (SMOOTH_TRIANGLE_TOKEN)
  2470.        Object = Parse_Smooth_Triangle ();
  2471.        EXIT
  2472.      END_CASE
  2473.  
  2474.      CASE (HEIGHT_FIELD_TOKEN)
  2475.        Object = Parse_Height_Field ();
  2476.        EXIT
  2477.      END_CASE
  2478.  
  2479.      CASE (BOX_TOKEN)
  2480.        Object = Parse_Box ();
  2481.        EXIT
  2482.      END_CASE
  2483.  
  2484.      CASE (BLOB_TOKEN)
  2485.        Object = Parse_Blob ();
  2486.        EXIT
  2487.      END_CASE
  2488.  
  2489.      CASE (LIGHT_SOURCE_TOKEN)
  2490.        Object = Parse_Light_Source ();
  2491.        EXIT
  2492.      END_CASE
  2493.  
  2494.      CASE (OBJECT_TOKEN)
  2495.        Parse_Begin ();
  2496.        Object = Parse_Object ();
  2497.        if (!Object)
  2498.          Parse_Error_Str ("object");
  2499.        Parse_Object_Mods ((OBJECT *)Object);
  2500.        EXIT
  2501.      END_CASE
  2502.  
  2503.      OTHERWISE
  2504.        UNGET
  2505.        EXIT
  2506.      END_CASE
  2507.    END_EXPECT
  2508.  
  2509.    return ((OBJECT *) Object);
  2510.   }
  2511.  
  2512. static void Parse_Fog ()
  2513.   {
  2514.    Parse_Begin ();
  2515.  
  2516.    EXPECT
  2517.      CASE (COLOUR_TOKEN)
  2518.        Parse_Colour (&Frame.Fog_Colour);
  2519.      END_CASE
  2520.  
  2521.      CASE (DISTANCE_TOKEN)
  2522.        Frame.Fog_Distance = Parse_Float ();
  2523.      END_CASE
  2524.  
  2525.      CASE_FLOAT
  2526.        Warn("Should use distance keyword.",1.5);
  2527.        Frame.Fog_Distance = Parse_Float ();
  2528.      END_CASE
  2529.  
  2530.      OTHERWISE
  2531.        UNGET
  2532.        EXIT
  2533.      END_CASE
  2534.    END_EXPECT
  2535.    Parse_End ();
  2536.   }
  2537.  
  2538. static void Parse_Frame ()
  2539.   {
  2540.    OBJECT *Object;
  2541.    TEXTURE *Local_Texture;
  2542.    PIGMENT *Local_Pigment;
  2543.    TNORMAL *Local_Tnormal;
  2544.    FINISH  *Local_Finish;
  2545.  
  2546.    EXPECT
  2547.      CASE (FOG_TOKEN)
  2548.        Parse_Fog();
  2549.      END_CASE
  2550.  
  2551.      CASE (BACKGROUND_TOKEN)
  2552.        Parse_Begin();
  2553.        GET (COLOUR_TOKEN)
  2554.        Parse_Colour (&Frame.Background_Colour);
  2555.        Parse_End();
  2556.      END_CASE
  2557.  
  2558.      CASE (CAMERA_TOKEN)
  2559.        Parse_Camera (&Frame.Camera);
  2560.      END_CASE
  2561.  
  2562.      CASE (DECLARE_TOKEN)
  2563.        Parse_Declare ();
  2564.      END_CASE
  2565.  
  2566.      CASE (MAX_TRACE_LEVEL_TOKEN)
  2567.        Max_Trace_Level = Parse_Float ();
  2568.      END_CASE
  2569.  
  2570.      CASE (VERSION_TOKEN)
  2571.        Language_Version = Parse_Float ();
  2572.      END_CASE
  2573.  
  2574.      CASE (MAX_INTERSECTIONS)
  2575.        Max_Intersections = (int)Parse_Float ();
  2576.      END_CASE
  2577.  
  2578.      CASE (DEFAULT_TOKEN)
  2579.        Not_In_Default = FALSE;
  2580.        Parse_Begin();
  2581.        EXPECT
  2582.          CASE (TEXTURE_TOKEN)
  2583.            Local_Texture = Default_Texture;
  2584.            Default_Texture = Parse_Texture();
  2585.            if (Default_Texture->Type != PNF_TEXTURE)
  2586.              Error("Default texture cannot be material map or tiles");
  2587.            if (Default_Texture->Next_Layer != NULL)
  2588.              Error("Default texture cannot be layered");
  2589.            Destroy_Textures(Local_Texture);
  2590.          END_CASE
  2591.  
  2592.          CASE (PIGMENT_TOKEN)
  2593.            Local_Pigment = Copy_Pigment((Default_Texture->Pigment));
  2594.            Parse_Pigment (&Local_Pigment);
  2595.            Destroy_Pigment(Default_Texture->Pigment);
  2596.            Default_Texture->Pigment = Local_Pigment;
  2597.          END_CASE
  2598.  
  2599.          CASE (TNORMAL_TOKEN)
  2600.            Local_Tnormal = Copy_Tnormal((Default_Texture->Tnormal));
  2601.            Parse_Tnormal (&Local_Tnormal);
  2602.            Destroy_Tnormal(Default_Texture->Tnormal);
  2603.            Default_Texture->Tnormal = Local_Tnormal;
  2604.          END_CASE
  2605.  
  2606.          CASE (FINISH_TOKEN)
  2607.            Local_Finish = Copy_Finish((Default_Texture->Finish));
  2608.            Parse_Finish (&Local_Finish);
  2609.            Destroy_Finish(Default_Texture->Finish);
  2610.            Default_Texture->Finish = Local_Finish;
  2611.          END_CASE
  2612.  
  2613.          CASE (CAMERA_TOKEN)
  2614.            Parse_Camera (&Default_Camera);
  2615.          END_CASE
  2616.  
  2617.          OTHERWISE
  2618.            UNGET
  2619.            EXIT
  2620.          END_CASE
  2621.        END_EXPECT
  2622.        Parse_End();
  2623.        Not_In_Default = TRUE;
  2624.      END_CASE
  2625.  
  2626.      CASE (END_OF_FILE_TOKEN)
  2627.        EXIT
  2628.      END_CASE
  2629.  
  2630.      OTHERWISE
  2631.        UNGET
  2632.        Object = Parse_Object();
  2633.        if (Object == NULL)
  2634.          Parse_Error_Str ("object or directive");
  2635.        Post_Process (Object, NULL);
  2636.        Link_To_Frame (Object);
  2637.      END_CASE
  2638.    END_EXPECT
  2639.   }
  2640.  
  2641. static void Parse_Camera (Camera_Ptr)
  2642.   CAMERA **Camera_Ptr;
  2643.   {
  2644.    VECTOR Local_Vector, Temp_Vector;
  2645.    DBL Direction_Length, Up_Length, Right_Length, Handedness;
  2646.    CAMERA *New;
  2647.  
  2648.    Parse_Begin ();
  2649.  
  2650.    EXPECT
  2651.      CASE (CAMERA_ID_TOKEN)
  2652.        Destroy_Camera(*Camera_Ptr);
  2653.        *Camera_Ptr = Copy_Camera ((CAMERA *) Token.Constant_Data);
  2654.        EXIT
  2655.      END_CASE
  2656.  
  2657.      OTHERWISE
  2658.        UNGET
  2659.        EXIT
  2660.      END_CASE
  2661.    END_EXPECT
  2662.  
  2663.    New = *Camera_Ptr;
  2664.  
  2665.    EXPECT
  2666.      CASE (LOCATION_TOKEN)
  2667.        Parse_Vector(&(New->Location));
  2668.      END_CASE
  2669.  
  2670.      CASE (DIRECTION_TOKEN)
  2671.        Parse_Vector(&(New->Direction));
  2672.      END_CASE
  2673.  
  2674.      CASE (UP_TOKEN)
  2675.        Parse_Vector(&(New->Up));
  2676.      END_CASE
  2677.  
  2678.      CASE (RIGHT_TOKEN)
  2679.        Parse_Vector(&(New->Right));
  2680.      END_CASE
  2681.  
  2682.      CASE (SKY_TOKEN)
  2683.        Parse_Vector(&(New->Sky));
  2684.      END_CASE
  2685.  
  2686.      CASE (LOOK_AT_TOKEN)
  2687.        VLength (Direction_Length, New->Direction);
  2688.        VLength (Up_Length,        New->Up);
  2689.        VLength (Right_Length,     New->Right);
  2690.        VCross  (Temp_Vector,      New->Direction, New->Up);
  2691.        VDot    (Handedness,       Temp_Vector,   New->Right);
  2692.  
  2693.        Parse_Vector (&New->Direction);
  2694.  
  2695.        VSub       (New->Direction, New->Direction, New->Location);
  2696.        VNormalize (New->Direction, New->Direction);
  2697.        VCross     (New->Right,     New->Direction, New->Sky);
  2698.        VNormalize (New->Right,     New->Right);
  2699.        VCross     (New->Up,        New->Right,     New->Direction);
  2700.        VScale     (New->Direction, New->Direction, Direction_Length);
  2701.  
  2702.        if (Handedness >= 0.0)
  2703.          VScale (New->Right, New->Right, Right_Length)
  2704.        else
  2705.          VScale (New->Right, New->Right, -Right_Length);
  2706.  
  2707.        VScale (New->Up, New->Up, Up_Length);
  2708.      END_CASE
  2709.  
  2710.      CASE (TRANSLATE_TOKEN)
  2711.        Parse_Vector (&Local_Vector);
  2712.        Translate_Camera (New, &Local_Vector);
  2713.      END_CASE
  2714.  
  2715.      CASE (ROTATE_TOKEN)
  2716.        Parse_Vector (&Local_Vector);
  2717.        Rotate_Camera (New, &Local_Vector);
  2718.      END_CASE
  2719.  
  2720.      CASE (SCALE_TOKEN)
  2721.        Parse_Scale_Vector (&Local_Vector);
  2722.        Scale_Camera (New, &Local_Vector);
  2723.      END_CASE
  2724.  
  2725.      CASE (TRANSFORM_TOKEN)
  2726.        GET(TRANSFORM_ID_TOKEN)
  2727.        Transform_Camera (New, (TRANSFORM *)Token.Constant_Data);
  2728.      END_CASE
  2729.  
  2730.      OTHERWISE
  2731.        UNGET
  2732.        EXIT
  2733.      END_CASE
  2734.    END_EXPECT
  2735.    Parse_End ();
  2736.   }
  2737.  
  2738. static
  2739. TRANSFORM *Parse_Transform ()
  2740.   {
  2741.    TRANSFORM *New, Local_Trans;
  2742.    VECTOR Local_Vector;
  2743.  
  2744.    Parse_Begin ();
  2745.    New = Create_Transform ();
  2746.  
  2747.    EXPECT
  2748.      CASE(TRANSFORM_ID_TOKEN)
  2749.        Compose_Transforms (New, (TRANSFORM *)Token.Constant_Data);
  2750.        EXIT
  2751.      END_CASE
  2752.  
  2753.      CASE (TRANSLATE_TOKEN)
  2754.        Parse_Vector (&Local_Vector);
  2755.        Compute_Translation_Transform(&Local_Trans, &Local_Vector);
  2756.        Compose_Transforms (New, &Local_Trans);
  2757.      END_CASE
  2758.  
  2759.      CASE (ROTATE_TOKEN)
  2760.        Parse_Vector (&Local_Vector);
  2761.        Compute_Rotation_Transform(&Local_Trans, &Local_Vector);
  2762.        Compose_Transforms (New, &Local_Trans);
  2763.      END_CASE
  2764.  
  2765.      CASE (SCALE_TOKEN)
  2766.        Parse_Scale_Vector (&Local_Vector);
  2767.        Compute_Scaling_Transform(&Local_Trans, &Local_Vector);
  2768.        Compose_Transforms (New, &Local_Trans);
  2769.      END_CASE
  2770.  
  2771.      OTHERWISE
  2772.        UNGET
  2773.        EXIT
  2774.      END_CASE
  2775.    END_EXPECT
  2776.  
  2777.    Parse_End ();
  2778.    return (New);
  2779.   }
  2780.  
  2781. static void Parse_Declare ()
  2782.   {
  2783.   VECTOR Local_Vector;
  2784.   COLOUR *Local_Colour;
  2785.   PIGMENT *Local_Pigment;
  2786.   TNORMAL *Local_Tnormal;
  2787.   FINISH *Local_Finish;
  2788.   TEXTURE *Local_Texture;
  2789.   COLOUR_MAP *Local_Colour_Map;
  2790.   TRANSFORM *Local_Trans;
  2791.   OBJECT *Local_Object;
  2792.   CAMERA *Local_Camera;
  2793.  
  2794.   struct Constant_Struct *Constant_Ptr;
  2795.  
  2796.   EXPECT
  2797.     CASE (IDENTIFIER_TOKEN)
  2798.       if (++Number_Of_Constants >= MAX_CONSTANTS)
  2799.         Error ("Too many constants \"DECLARED\"");
  2800.       else
  2801.         Constant_Ptr = &(Constants[Number_Of_Constants]);
  2802.       EXIT
  2803.     END_CASE
  2804.  
  2805.     CASE4 (COLOUR_ID_TOKEN, VECTOR_ID_TOKEN, FLOAT_ID_TOKEN, PIGMENT_ID_TOKEN)
  2806.     CASE4 (TNORMAL_ID_TOKEN, FINISH_ID_TOKEN, TEXTURE_ID_TOKEN, OBJECT_ID_TOKEN)
  2807.     CASE3 (COLOUR_MAP_ID_TOKEN, TRANSFORM_ID_TOKEN, CAMERA_ID_TOKEN)
  2808.       Constant_Ptr = &(Constants[Token.Constant_Index]);
  2809.       EXIT
  2810.     END_CASE
  2811.  
  2812.     OTHERWISE
  2813.       Parse_Error(IDENTIFIER_TOKEN);
  2814.     END_CASE
  2815.   END_EXPECT
  2816.  
  2817.   Previous = Token.Token_Id;
  2818.  
  2819.   GET (EQUALS_TOKEN);
  2820.  
  2821.   EXPECT
  2822.     CASE (COLOUR_TOKEN)
  2823.       if (Test_Redefine(COLOUR_ID_TOKEN))
  2824.         Destroy_Colour((COLOUR *)Constant_Ptr->Constant_Data);
  2825.       Local_Colour = Create_Colour();
  2826.       Parse_Colour (Local_Colour);
  2827.       Constant_Ptr -> Constant_Data = (char *) Local_Colour;
  2828.       Constant_Ptr -> Constant_Type = COLOUR_CONSTANT;
  2829.       EXIT
  2830.     END_CASE
  2831.  
  2832.     CASE_VECTOR
  2833.       Have_Vector = FALSE;
  2834.       Parse_Vector_Float (&Local_Vector);
  2835.       if (Have_Vector)
  2836.         {
  2837.          if (Test_Redefine(VECTOR_ID_TOKEN))
  2838.            Destroy_Vector((VECTOR *)Constant_Ptr->Constant_Data);
  2839.          Constant_Ptr -> Constant_Type = VECTOR_CONSTANT;
  2840.          Constant_Ptr -> Constant_Data = (char *) Create_Vector();
  2841.          *((VECTOR *)Constant_Ptr -> Constant_Data) = Local_Vector;
  2842.         }
  2843.       else
  2844.         {
  2845.          if (Test_Redefine(FLOAT_ID_TOKEN))
  2846.            Destroy_Float((DBL *)Constant_Ptr->Constant_Data);
  2847.          Constant_Ptr -> Constant_Type = FLOAT_CONSTANT;
  2848.          Constant_Ptr -> Constant_Data = (char *) Create_Float();
  2849.          *((DBL *) Constant_Ptr -> Constant_Data) = Local_Vector.x;
  2850.         }
  2851.       EXIT
  2852.     END_CASE
  2853.  
  2854.     CASE (PIGMENT_TOKEN)
  2855.       if (Test_Redefine(PIGMENT_ID_TOKEN))
  2856.         Destroy_Pigment((PIGMENT *)Constant_Ptr->Constant_Data);
  2857.       Local_Pigment = Copy_Pigment((Default_Texture->Pigment));
  2858.       Parse_Pigment (&Local_Pigment);
  2859.       Constant_Ptr -> Constant_Type = PIGMENT_CONSTANT;
  2860.       Constant_Ptr -> Constant_Data = (char *)Local_Pigment;
  2861.       EXIT
  2862.     END_CASE
  2863.  
  2864.     CASE (TNORMAL_TOKEN)
  2865.       if (Test_Redefine(TNORMAL_ID_TOKEN))
  2866.         Destroy_Tnormal((TNORMAL *)Constant_Ptr->Constant_Data);
  2867.       Local_Tnormal = Copy_Tnormal((Default_Texture->Tnormal));
  2868.       Parse_Tnormal (&Local_Tnormal);
  2869.       Constant_Ptr -> Constant_Type = TNORMAL_CONSTANT;
  2870.       Constant_Ptr -> Constant_Data = (char *) Local_Tnormal;
  2871.       EXIT
  2872.     END_CASE
  2873.  
  2874.     CASE (FINISH_TOKEN)
  2875.       if (Test_Redefine(FINISH_ID_TOKEN))
  2876.         Destroy_Finish((FINISH *)Constant_Ptr->Constant_Data);
  2877.       Local_Finish = Copy_Finish((Default_Texture->Finish));
  2878.       Parse_Finish (&Local_Finish);
  2879.       Constant_Ptr -> Constant_Type = FINISH_CONSTANT;
  2880.       Constant_Ptr -> Constant_Data = (char *) Local_Finish;
  2881.       EXIT
  2882.     END_CASE
  2883.  
  2884.     CASE (CAMERA_TOKEN)
  2885.       if (Test_Redefine(CAMERA_ID_TOKEN))
  2886.         Destroy_Camera((CAMERA *)Constant_Ptr->Constant_Data);
  2887.       Local_Camera = Copy_Camera(Default_Camera);
  2888.       Parse_Camera (&Local_Camera);
  2889.       Constant_Ptr -> Constant_Type = CAMERA_CONSTANT;
  2890.       Constant_Ptr -> Constant_Data = (char *) Local_Camera;
  2891.       EXIT
  2892.     END_CASE
  2893.  
  2894.     CASE (TEXTURE_TOKEN)
  2895.       if (Test_Redefine(TEXTURE_ID_TOKEN))
  2896.         Destroy_Textures((TEXTURE *)Constant_Ptr->Constant_Data);
  2897.       Local_Texture = Parse_Texture ();
  2898.       Constant_Ptr -> Constant_Type = TEXTURE_CONSTANT;
  2899.       Constant_Ptr -> Constant_Data = NULL;
  2900.       Link_Textures((TEXTURE **) &Constant_Ptr->Constant_Data, Local_Texture);
  2901.       EXPECT
  2902.         CASE (TEXTURE_TOKEN)
  2903.           Local_Texture = Parse_Texture ();
  2904.           Link_Textures((TEXTURE **) &Constant_Ptr->Constant_Data, Local_Texture);
  2905.         END_CASE
  2906.  
  2907.         OTHERWISE
  2908.           UNGET
  2909.           EXIT
  2910.         END_CASE
  2911.       END_EXPECT
  2912.       EXIT
  2913.     END_CASE
  2914.  
  2915.     CASE (COLOUR_MAP_TOKEN)
  2916.       if (Test_Redefine(COLOUR_MAP_ID_TOKEN))
  2917.         Destroy_Colour_Map((COLOUR_MAP *)Constant_Ptr->Constant_Data);
  2918.       Local_Colour_Map = Parse_Colour_Map ();
  2919.       Constant_Ptr -> Constant_Type = COLOUR_MAP_CONSTANT;
  2920.       Constant_Ptr -> Constant_Data = (char *) Local_Colour_Map;
  2921.       EXIT
  2922.     END_CASE
  2923.  
  2924.     CASE (TRANSFORM_TOKEN)
  2925.       if (Test_Redefine(TRANSFORM_ID_TOKEN))
  2926.         Destroy_Transform((TRANSFORM *)Constant_Ptr->Constant_Data);
  2927.       Local_Trans = Parse_Transform ();
  2928.       Constant_Ptr -> Constant_Type = TRANSFORM_CONSTANT;
  2929.       Constant_Ptr -> Constant_Data = (char *) Local_Trans;
  2930.       EXIT
  2931.     END_CASE
  2932.  
  2933.     OTHERWISE
  2934.       UNGET
  2935.       if (Test_Redefine(OBJECT_ID_TOKEN))
  2936.         Destroy_Object((OBJECT *)Constant_Ptr->Constant_Data);
  2937.       Local_Object = Parse_Object ();
  2938.       Constant_Ptr -> Constant_Type = OBJECT_CONSTANT;
  2939.       Constant_Ptr -> Constant_Data = (char *) Local_Object;
  2940.       EXIT
  2941.     END_CASE
  2942.  
  2943.   END_EXPECT
  2944.   }
  2945.  
  2946. static void Link (New_Object, Field, Old_Object_List)
  2947.   OBJECT *New_Object, **Field, **Old_Object_List;
  2948.   {
  2949.   *Field = *Old_Object_List;
  2950.   *Old_Object_List = New_Object;
  2951.   }
  2952.  
  2953. static void Link_Textures (Old_Textures, New_Textures)
  2954.   TEXTURE **Old_Textures;
  2955.   TEXTURE  *New_Textures;
  2956.   {
  2957.    TEXTURE *Layer;
  2958.  
  2959.    for (Layer = New_Textures ;
  2960.         Layer->Next_Layer != NULL ;
  2961.         Layer = Layer->Next_Layer)
  2962.      {}
  2963.         Layer->Next_Layer = *Old_Textures;
  2964.         *Old_Textures = New_Textures;
  2965.   }
  2966.  
  2967. static
  2968. char *Get_Token_String (Token_Id)
  2969.   TOKEN Token_Id;
  2970.   {
  2971.   register int i;
  2972.  
  2973.   for (i = 0 ; i < LAST_TOKEN ; i++)
  2974.      if (Reserved_Words[i].Token_Number == Token_Id)
  2975.         return (Reserved_Words[i].Token_Name);
  2976.   return ("");
  2977.   }
  2978.  
  2979. static
  2980. void Where_Error ()
  2981.   {
  2982.   fprintf (stderr, "\nError in file %s line %d\n", Token.Filename,
  2983.                                                  Token.Token_Line_No+1);
  2984.   }
  2985.  
  2986. static int Test_Redefine(a)
  2987.   int a;
  2988.   {
  2989.   char *old, *new;
  2990.  
  2991.   if (Previous == IDENTIFIER_TOKEN)
  2992.     return (FALSE);
  2993.   if (Previous != a)
  2994.     {old = Get_Token_String (Previous);
  2995.      new = Get_Token_String (a);
  2996.      Where_Error ();
  2997.      fprintf (stderr, "Attempted to redefine %s as %s", old, new);
  2998.      exit (1);
  2999.     }
  3000.   return (TRUE);
  3001.   }
  3002.  
  3003. void Parse_Error (Token_Id)
  3004.   TOKEN Token_Id;
  3005.   {
  3006.   char *expected;
  3007.  
  3008.   expected = Get_Token_String (Token_Id);
  3009.   Parse_Error_Str(expected);
  3010.   }
  3011.  
  3012. void Parse_Error_Str (str)
  3013.   char *str;
  3014.   {
  3015.    Where_Error ();
  3016.    fprintf (stderr, "%s expected but", str);
  3017.    Found_Instead ();
  3018.    exit (1);
  3019.   }
  3020.  
  3021. static void Found_Instead ()
  3022.   {
  3023.   char *found;
  3024.  
  3025.   if (Token.Token_Id == IDENTIFIER_TOKEN)
  3026.     fprintf (stderr,
  3027.       " undeclared identifier '%s' found instead.\n", Token.Token_String);
  3028.   else
  3029.    {
  3030.     found = Get_Token_String (Token.Token_Id);
  3031.     fprintf (stderr, " %s found instead.\n", found);
  3032.    }
  3033.   }
  3034. /*
  3035. static void Parse_Warn (Token_Id)
  3036.   TOKEN Token_Id;
  3037.   {
  3038.   char *expected;
  3039.  
  3040.   fprintf (stderr, "\nWarning in file %s line %d\n", Token.Filename,
  3041.                                                    Token.Token_Line_No+1);
  3042.   expected = Get_Token_String (Token_Id);
  3043.   fprintf (stderr, "%s expected but", expected);
  3044.   Found_Instead ();
  3045.   }
  3046. */
  3047. static void Warn_State (Token_Id,Type)
  3048.   TOKEN Token_Id, Type;
  3049.   {
  3050.   char *found;
  3051.   char *should;
  3052.  
  3053.   if (Language_Version < 1.5)
  3054.      return;
  3055.  
  3056.   fprintf (stderr, "\nWarning in file %s line %d\n", Token.Filename,
  3057.                                                    Token.Token_Line_No+1);
  3058.   found = Get_Token_String (Token_Id);
  3059.   should = Get_Token_String (Type);
  3060.   fprintf (stderr, "Found %s that should be in %s statement", found, should);
  3061.   }
  3062.  
  3063. void Warn (str,Level)
  3064.   char *str;
  3065.   DBL Level;
  3066.   {
  3067.   if (Language_Version < Level)
  3068.     return;
  3069.     
  3070.   fprintf (stdout, "\nWarning in file %s line %d\n", Token.Filename,
  3071.                                                    Token.Token_Line_No+1);
  3072.   fputs (str, stdout);
  3073.   }
  3074.  
  3075. void Error (str)
  3076.   char *str;
  3077.   {
  3078.   Where_Error ();
  3079.   fputs (str, stderr);
  3080.   exit (1);
  3081.   }
  3082.  
  3083. void MAError (str)
  3084.   char *str;
  3085.   {
  3086.   Where_Error ();
  3087.   fprintf (stderr, "Out of memory.  Cannot allocate %s.\n",str);
  3088.   exit (1);
  3089.   }
  3090.  
  3091. /* Write a token out to the token file */
  3092.  
  3093. void Write_Token (Token_Id, Data_File)
  3094.   TOKEN Token_Id;
  3095.   DATA_FILE *Data_File;
  3096.  
  3097.   {
  3098.    Token.Token_Id = Token_Id;
  3099.    Token.Token_Line_No = Data_File->Line_Number;
  3100.    Token.Filename = Data_File->Filename;
  3101.    Token.Token_String = String;
  3102.    Token.Constant_Data = NULL;
  3103.    Token.Constant_Index = (int) Token.Token_Id - (int) LAST_TOKEN;
  3104.  
  3105.    if (Token.Constant_Index >= 0)
  3106.      {if (Token.Constant_Index <= Number_Of_Constants)
  3107.         {Token.Constant_Data = Constants[Token.Constant_Index].Constant_Data;
  3108.          switch (Constants[Token.Constant_Index].Constant_Type)
  3109.            {CASEID(COLOUR_CONSTANT,     COLOUR_ID_TOKEN)
  3110.             CASEID(VECTOR_CONSTANT,     VECTOR_ID_TOKEN)
  3111.             CASEID(FLOAT_CONSTANT,      FLOAT_ID_TOKEN)
  3112.             CASEID(PIGMENT_CONSTANT,    PIGMENT_ID_TOKEN)
  3113.             CASEID(TNORMAL_CONSTANT,    TNORMAL_ID_TOKEN)
  3114.             CASEID(FINISH_CONSTANT,     FINISH_ID_TOKEN)
  3115.             CASEID(TEXTURE_CONSTANT,    TEXTURE_ID_TOKEN)
  3116.             CASEID(OBJECT_CONSTANT,     OBJECT_ID_TOKEN)
  3117.             CASEID(COLOUR_MAP_CONSTANT, COLOUR_MAP_ID_TOKEN)
  3118.             CASEID(TRANSFORM_CONSTANT,  TRANSFORM_ID_TOKEN)
  3119.             CASEID(CAMERA_CONSTANT,     CAMERA_ID_TOKEN)
  3120.            }
  3121.         }
  3122.       else Token.Token_Id = IDENTIFIER_TOKEN;
  3123.      }
  3124.   }
  3125.  
  3126. static void Post_Process (Object,Parent)
  3127.   OBJECT *Object, *Parent;
  3128.   {
  3129.    OBJECT *Sib;
  3130.  
  3131.    if (Object == NULL)
  3132.      return;
  3133.  
  3134.    if (Parent != NULL)
  3135.      {
  3136.       if (Object->Texture == NULL)
  3137.         Object->Texture = Parent->Texture;
  3138. /*
  3139.       else
  3140.         if (Parent->Texture != NULL)
  3141.           {Local_Texture = Copy_Textures (Parent->Texture);
  3142.            Link_Textures (&(Object->Texture), Local_Texture);
  3143.           }
  3144. */ /* Removed for backward compat with 1.0.  May put back in. CEY 12/92 */
  3145.       Object->No_Shadow_Flag |= Parent->No_Shadow_Flag;
  3146.      }
  3147.      
  3148.    if (     (Object->Texture == NULL) 
  3149.         && !(Object->Type & TEXTURED_OBJECT) 
  3150.         && !(Object->Type & LIGHT_SOURCE_OBJECT))
  3151.      Object->Texture = Copy_Textures(Default_Texture);
  3152.  
  3153.    if (Object->Type & COMPOUND_OBJECT)
  3154.      {
  3155.       if (Object->Type & LIGHT_SOURCE_OBJECT)
  3156.         {
  3157.          ((LIGHT_SOURCE *)Object)->Next_Light_Source = Frame.Light_Sources;
  3158.          Frame.Light_Sources = (LIGHT_SOURCE *)Object;
  3159.         }
  3160.       for (Sib = ((CSG *)Object)->Children;
  3161.            Sib != NULL;
  3162.            Sib = Sib->Sibling)
  3163.         Post_Process(Sib, Object);
  3164.      }
  3165.    else
  3166.      if (Object->Texture == NULL)
  3167.        Object->Texture = Copy_Textures(Default_Texture);
  3168.    Post_Textures (Object->Texture);
  3169.    if ((Object->Type & WATER_LEVEL_OK_OBJECT) &&
  3170.        (Object->Type & IS_CHILD_OBJECT))
  3171.      Object->Methods = &Csg_Height_Field_Methods;
  3172.   }
  3173.  
  3174. static void Destroy_Constants ()
  3175.   {
  3176.    int i;
  3177.    char *Ptr;
  3178.  
  3179.    for (i=1; i <= Number_Of_Constants; i++)
  3180.      {
  3181.       Ptr = Constants[i].Constant_Data;
  3182.       switch (Constants[i].Constant_Type)
  3183.         {
  3184.          case COLOUR_CONSTANT:
  3185.            Destroy_Colour((COLOUR *)Ptr);
  3186.            break;
  3187.          case VECTOR_CONSTANT:
  3188.            Destroy_Vector((VECTOR *)Ptr);
  3189.            break;
  3190.          case FLOAT_CONSTANT:
  3191.            Destroy_Float((DBL *)Ptr);
  3192.            break;
  3193.          case PIGMENT_CONSTANT:
  3194.            Destroy_Pigment((PIGMENT *)Ptr);
  3195.            break;
  3196.          case TNORMAL_CONSTANT:
  3197.            Destroy_Tnormal((TNORMAL *)Ptr);
  3198.            break;
  3199.          case FINISH_CONSTANT:
  3200.            Destroy_Finish((FINISH *)Ptr);
  3201.            break;
  3202.          case TEXTURE_CONSTANT:
  3203.            Destroy_Textures((TEXTURE *)Ptr);
  3204.            break;
  3205.          case OBJECT_CONSTANT:
  3206.            Destroy_Object((OBJECT *)Ptr);
  3207.            break;
  3208.          case COLOUR_MAP_CONSTANT:
  3209.            Destroy_Colour_Map((COLOUR_MAP *)Ptr);
  3210.            break;
  3211.          case TRANSFORM_CONSTANT:
  3212.            Destroy_Transform((TRANSFORM *)Ptr);
  3213.            break;
  3214.          case CAMERA_CONSTANT:
  3215.            Destroy_Camera((CAMERA *)Ptr);
  3216.            break;
  3217.         }
  3218.      }
  3219.   }
  3220.  
  3221. static void Link_To_Frame (Object)
  3222.  OBJECT *Object;
  3223.  {
  3224.   OBJECT *This_Sib, *Next_Sib;
  3225.   
  3226.   if ((Object->Methods != &CSG_Union_Methods) ||
  3227.       (Object->Bound != NULL) ||
  3228.       (Object->Clip != NULL) ||
  3229.       (!Use_Slabs))
  3230.     {
  3231.      Link(Object, &(Object -> Sibling), &(Frame.Objects));
  3232.      return;
  3233.     }
  3234.   
  3235.   for (This_Sib = ((CSG *)Object)->Children;
  3236.        This_Sib != NULL;
  3237.        This_Sib = Next_Sib)
  3238.        {
  3239.         Next_Sib = This_Sib->Sibling; /*L2F changes Sibling so save it */
  3240.         Link_To_Frame (This_Sib);
  3241.        }
  3242.   Object->Texture = NULL;
  3243.   Object->Sibling = NULL;
  3244.   ((CSG *)Object)->Children = NULL;
  3245.   Destroy_Object (Object);
  3246.  }
  3247.