Annotation of GNUtools/cc/mips-tfile.c, revision 1.1.1.1

1.1       root        1: /* Update the symbol table (the .T file) in a MIPS object to
                      2:    contain debugging information specified by the GNU compiler
                      3:    in the form of comments (the mips assembler does not support
                      4:    assembly access to debug information).
                      5:    Contributed by:  Michael Meissner, [email protected]
                      6:    Copyright (C) 1991, 1993 Free Software Foundation, Inc.
                      7: 
                      8: This file is part of GNU CC.
                      9: 
                     10: GNU CC is free software; you can redistribute it and/or modify
                     11: it under the terms of the GNU General Public License as published by
                     12: the Free Software Foundation; either version 2, or (at your option)
                     13: any later version.
                     14: 
                     15: GNU CC is distributed in the hope that it will be useful,
                     16: but WITHOUT ANY WARRANTY; without even the implied warranty of
                     17: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
                     18: GNU General Public License for more details.
                     19: 
                     20: You should have received a copy of the GNU General Public License
                     21: along with GNU CC; see the file COPYING.  If not, write to
                     22: the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
                     23: 
                     24: 
                     25: /* Here is a brief description of the MIPS ECOFF symbol table.  The
                     26:    MIPS symbol table has the following pieces:
                     27: 
                     28:        Symbolic Header
                     29:            |
                     30:            +-- Auxiliary Symbols
                     31:            |
                     32:            +-- Dense number table
                     33:            |
                     34:            +-- Optimizer Symbols
                     35:            |
                     36:            +-- External Strings
                     37:            |
                     38:            +-- External Symbols
                     39:            |
                     40:            +-- Relative file descriptors
                     41:            |
                     42:            +-- File table
                     43:                    |
                     44:                    +-- Procedure table
                     45:                    |
                     46:                    +-- Line number table
                     47:                    |
                     48:                    +-- Local Strings
                     49:                    |
                     50:                    +-- Local Symbols
                     51: 
                     52:    The symbolic header points to each of the other tables, and also
                     53:    contains the number of entries.  It also contains a magic number
                     54:    and MIPS compiler version number, such as 2.0.
                     55: 
                     56:    The auxiliary table is a series of 32 bit integers, that are
                     57:    referenced as needed from the local symbol table.  Unlike standard
                     58:    COFF, the aux.  information does not follow the symbol that uses
                     59:    it, but rather is a separate table.  In theory, this would allow
                     60:    the MIPS compilers to collapse duplicate aux. entries, but I've not
                     61:    noticed this happening with the 1.31 compiler suite.  The different
                     62:    types of aux. entries are:
                     63: 
                     64:     1) dnLow: Low bound on array dimension.
                     65: 
                     66:     2) dnHigh: High bound on array dimension.
                     67: 
                     68:     3) isym: Index to the local symbol which is the start of the
                     69:        function for the end of function first aux. entry.
                     70: 
                     71:     4) width: Width of structures and bitfields.
                     72: 
                     73:     5) count: Count of ranges for variant part.
                     74: 
                     75:     6) rndx: A relative index into the symbol table.  The relative
                     76:        index field has two parts: rfd which is a pointer into the
                     77:        relative file index table or ST_RFDESCAPE which says the next
                     78:        aux. entry is the file number, and index: which is the pointer
                     79:        into the local symbol within a given file table.  This is for
                     80:        things like references to types defined in another file.
                     81: 
                     82:     7) Type information: This is like the COFF type bits, except it
                     83:        is 32 bits instead of 16; they still have room to add new
                     84:        basic types; and they can handle more than 6 levels of array,
                     85:        pointer, function, etc.  Each type information field contains
                     86:        the following structure members:
                     87: 
                     88:            a)  fBitfield: a bit that says this is a bitfield, and the
                     89:                size in bits follows as the next aux. entry.
                     90: 
                     91:            b)  continued: a bit that says the next aux. entry is a
                     92:                continuation of the current type information (in case
                     93:                there are more than 6 levels of array/ptr/function).
                     94: 
                     95:            c)  bt: an integer containing the base type before adding
                     96:                array, pointer, function, etc. qualifiers.  The
                     97:                current base types that I have documentation for are:
                     98: 
                     99:                        btNil           -- undefined 
                    100:                        btAdr           -- address - integer same size as ptr
                    101:                        btChar          -- character 
                    102:                        btUChar         -- unsigned character 
                    103:                        btShort         -- short 
                    104:                        btUShort        -- unsigned short 
                    105:                        btInt           -- int 
                    106:                        btUInt          -- unsigned int 
                    107:                        btLong          -- long 
                    108:                        btULong         -- unsigned long 
                    109:                        btFloat         -- float (real) 
                    110:                        btDouble        -- Double (real) 
                    111:                        btStruct        -- Structure (Record) 
                    112:                        btUnion         -- Union (variant) 
                    113:                        btEnum          -- Enumerated 
                    114:                        btTypedef       -- defined via a typedef isymRef 
                    115:                        btRange         -- subrange of int 
                    116:                        btSet           -- pascal sets 
                    117:                        btComplex       -- fortran complex 
                    118:                        btDComplex      -- fortran double complex 
                    119:                        btIndirect      -- forward or unnamed typedef 
                    120:                        btFixedDec      -- Fixed Decimal 
                    121:                        btFloatDec      -- Float Decimal 
                    122:                        btString        -- Varying Length Character String 
                    123:                        btBit           -- Aligned Bit String 
                    124:                        btPicture       -- Picture
                    125:                        btVoid          -- Void (MIPS cc revision >= 2.00)
                    126: 
                    127:            d)  tq0 - tq5: type qualifier fields as needed.  The
                    128:                current type qualifier fields I have documentation for
                    129:                are:
                    130: 
                    131:                        tqNil           -- no more qualifiers 
                    132:                        tqPtr           -- pointer 
                    133:                        tqProc          -- procedure 
                    134:                        tqArray         -- array 
                    135:                        tqFar           -- 8086 far pointers 
                    136:                        tqVol           -- volatile 
                    137: 
                    138: 
                    139:    The dense number table is used in the front ends, and disappears by
                    140:    the time the .o is created.
                    141: 
                    142:    With the 1.31 compiler suite, the optimization symbols don't seem
                    143:    to be used as far as I can tell.
                    144: 
                    145:    The linker is the first entity that creates the relative file
                    146:    descriptor table, and I believe it is used so that the individual
                    147:    file table pointers don't have to be rewritten when the objects are
                    148:    merged together into the program file.
                    149: 
                    150:    Unlike COFF, the basic symbol & string tables are split into
                    151:    external and local symbols/strings.  The relocation information
                    152:    only goes off of the external symbol table, and the debug
                    153:    information only goes off of the internal symbol table.  The
                    154:    external symbols can have links to an appropriate file index and
                    155:    symbol within the file to give it the appropriate type information.
                    156:    Because of this, the external symbols are actually larger than the
                    157:    internal symbols (to contain the link information), and contain the
                    158:    local symbol structure as a member, though this member is not the
                    159:    first member of the external symbol structure (!).  I suspect this
                    160:    split is to make strip easier to deal with.
                    161: 
                    162:    Each file table has offsets for where the line numbers, local
                    163:    strings, local symbols, and procedure table starts from within the
                    164:    global tables, and the indexs are reset to 0 for each of those
                    165:    tables for the file.
                    166: 
                    167:    The procedure table contains the binary equivalents of the .ent
                    168:    (start of the function address), .frame (what register is the
                    169:    virtual frame pointer, constant offset from the register to obtain
                    170:    the VFP, and what register holds the return address), .mask/.fmask
                    171:    (bitmask of saved registers, and where the first register is stored
                    172:    relative to the VFP) assembler directives.  It also contains the
                    173:    low and high bounds of the line numbers if debugging is turned on.
                    174: 
                    175:    The line number table is a compressed form of the normal COFF line
                    176:    table.  Each line number entry is either 1 or 3 bytes long, and
                    177:    contains a signed delta from the previous line, and an unsigned
                    178:    count of the number of instructions this statement takes.
                    179: 
                    180:    The local symbol table contains the following fields:
                    181: 
                    182:     1) iss: index to the local string table giving the name of the
                    183:        symbol.
                    184: 
                    185:     2) value: value of the symbol (address, register number, etc.).
                    186: 
                    187:     3) st: symbol type.  The current symbol types are:
                    188: 
                    189:            stNil         -- Nuthin' special
                    190:            stGlobal      -- external symbol
                    191:            stStatic      -- static
                    192:            stParam       -- procedure argument
                    193:            stLocal       -- local variable
                    194:            stLabel       -- label
                    195:            stProc        -- External Procedure
                    196:            stBlock       -- beginning of block
                    197:            stEnd         -- end (of anything)
                    198:            stMember      -- member (of anything)
                    199:            stTypedef     -- type definition
                    200:            stFile        -- file name
                    201:            stRegReloc    -- register relocation
                    202:            stForward     -- forwarding address
                    203:            stStaticProc  -- Static procedure
                    204:            stConstant    -- const
                    205: 
                    206:     4) sc: storage class.  The current storage classes are:
                    207: 
                    208:            scText        -- text symbol
                    209:            scData        -- initialized data symbol
                    210:            scBss         -- un-initialized data symbol
                    211:            scRegister    -- value of symbol is register number
                    212:            scAbs         -- value of symbol is absolute
                    213:            scUndefined   -- who knows?
                    214:            scCdbLocal    -- variable's value is IN se->va.??
                    215:            scBits        -- this is a bit field
                    216:            scCdbSystem   -- value is IN debugger's address space
                    217:            scRegImage    -- register value saved on stack
                    218:            scInfo        -- symbol contains debugger information
                    219:            scUserStruct  -- addr in struct user for current process
                    220:            scSData       -- load time only small data
                    221:            scSBss        -- load time only small common
                    222:            scRData       -- load time only read only data
                    223:            scVar         -- Var parameter (fortranpascal)
                    224:            scCommon      -- common variable
                    225:            scSCommon     -- small common
                    226:            scVarRegister -- Var parameter in a register
                    227:            scVariant     -- Variant record
                    228:            scSUndefined  -- small undefined(external) data
                    229:            scInit        -- .init section symbol
                    230: 
                    231:     5) index: pointer to a local symbol or aux. entry.
                    232: 
                    233: 
                    234: 
                    235:    For the following program:
                    236: 
                    237:        #include <stdio.h>
                    238: 
                    239:        main(){
                    240:                printf("Hello World!\n");
                    241:                return 0;
                    242:        }
                    243: 
                    244:    Mips-tdump produces the following information:
                    245:    
                    246:    Global file header:
                    247:        magic number             0x162
                    248:        # sections               2
                    249:        timestamp                645311799, Wed Jun 13 17:16:39 1990
                    250:        symbolic header offset   284
                    251:        symbolic header size     96
                    252:        optional header          56
                    253:        flags                    0x0
                    254:    
                    255:    Symbolic header, magic number = 0x7009, vstamp = 1.31:
                    256:    
                    257:        Info                      Offset      Number       Bytes
                    258:        ====                      ======      ======      =====
                    259:    
                    260:        Line numbers                 380           4           4 [13]
                    261:        Dense numbers                  0           0           0
                    262:        Procedures Tables            384           1          52
                    263:        Local Symbols                436          16         192
                    264:        Optimization Symbols           0           0           0
                    265:        Auxiliary Symbols            628          39         156
                    266:        Local Strings                784          80          80
                    267:        External Strings             864         144         144
                    268:        File Tables                 1008           2         144
                    269:        Relative Files                 0           0           0
                    270:        External Symbols            1152          20         320
                    271:    
                    272:    File #0, "hello2.c"
                    273:    
                    274:        Name index  = 1          Readin      = No
                    275:        Merge       = No         Endian      = LITTLE
                    276:        Debug level = G2         Language    = C
                    277:        Adr         = 0x00000000
                    278:    
                    279:        Info                       Start      Number        Size      Offset
                    280:        ====                       =====      ======        ====      ======
                    281:        Local strings                  0          15          15         784
                    282:        Local symbols                  0           6          72         436
                    283:        Line numbers                   0          13          13         380
                    284:        Optimization symbols           0           0           0           0
                    285:        Procedures                     0           1          52         384
                    286:        Auxiliary symbols              0          14          56         628
                    287:        Relative Files                 0           0           0           0
                    288:    
                    289:     There are 6 local symbols, starting at 436
                    290: 
                    291:        Symbol# 0: "hello2.c"
                    292:            End+1 symbol  = 6
                    293:            String index  = 1
                    294:            Storage class = Text        Index  = 6
                    295:            Symbol type   = File        Value  = 0
                    296: 
                    297:        Symbol# 1: "main"
                    298:            End+1 symbol  = 5
                    299:            Type          = int
                    300:            String index  = 10
                    301:            Storage class = Text        Index  = 12
                    302:            Symbol type   = Proc        Value  = 0
                    303: 
                    304:        Symbol# 2: ""
                    305:            End+1 symbol  = 4
                    306:            String index  = 0
                    307:            Storage class = Text        Index  = 4
                    308:            Symbol type   = Block       Value  = 8
                    309: 
                    310:        Symbol# 3: ""
                    311:            First symbol  = 2
                    312:            String index  = 0
                    313:            Storage class = Text        Index  = 2
                    314:            Symbol type   = End         Value  = 28
                    315: 
                    316:        Symbol# 4: "main"
                    317:            First symbol  = 1
                    318:            String index  = 10
                    319:            Storage class = Text        Index  = 1
                    320:            Symbol type   = End         Value  = 52
                    321: 
                    322:        Symbol# 5: "hello2.c"
                    323:            First symbol  = 0
                    324:            String index  = 1
                    325:            Storage class = Text        Index  = 0
                    326:            Symbol type   = End         Value  = 0
                    327: 
                    328:     There are 14 auxiliary table entries, starting at 628.
                    329: 
                    330:        * #0               0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    331:        * #1              24, [  24/      0], [ 6 0:0 0:0:0:0:0:0]
                    332:        * #2               8, [   8/      0], [ 2 0:0 0:0:0:0:0:0]
                    333:        * #3              16, [  16/      0], [ 4 0:0 0:0:0:0:0:0]
                    334:        * #4              24, [  24/      0], [ 6 0:0 0:0:0:0:0:0]
                    335:        * #5              32, [  32/      0], [ 8 0:0 0:0:0:0:0:0]
                    336:        * #6              40, [  40/      0], [10 0:0 0:0:0:0:0:0]
                    337:        * #7              44, [  44/      0], [11 0:0 0:0:0:0:0:0]
                    338:        * #8              12, [  12/      0], [ 3 0:0 0:0:0:0:0:0]
                    339:        * #9              20, [  20/      0], [ 5 0:0 0:0:0:0:0:0]
                    340:        * #10             28, [  28/      0], [ 7 0:0 0:0:0:0:0:0]
                    341:        * #11             36, [  36/      0], [ 9 0:0 0:0:0:0:0:0]
                    342:          #12              5, [   5/      0], [ 1 1:0 0:0:0:0:0:0]
                    343:          #13             24, [  24/      0], [ 6 0:0 0:0:0:0:0:0]
                    344: 
                    345:     There are 1 procedure descriptor entries, starting at 0.
                    346: 
                    347:        Procedure descriptor 0:
                    348:            Name index   = 10          Name          = "main"
                    349:            .mask 0x80000000,-4        .fmask 0x00000000,0
                    350:            .frame $29,24,$31
                    351:            Opt. start   = -1          Symbols start = 1
                    352:            First line # = 3           Last line #   = 6
                    353:            Line Offset  = 0           Address       = 0x00000000
                    354: 
                    355:        There are 4 bytes holding line numbers, starting at 380.
                    356:            Line           3,   delta     0,   count  2
                    357:            Line           4,   delta     1,   count  3
                    358:            Line           5,   delta     1,   count  2
                    359:            Line           6,   delta     1,   count  6
                    360: 
                    361:    File #1, "/usr/include/stdio.h"
                    362: 
                    363:     Name index  = 1          Readin      = No
                    364:     Merge       = Yes        Endian      = LITTLE
                    365:     Debug level = G2         Language    = C
                    366:     Adr         = 0x00000000
                    367: 
                    368:     Info                       Start      Number        Size      Offset
                    369:     ====                       =====      ======        ====      ======
                    370:     Local strings                 15          65          65         799
                    371:     Local symbols                  6          10         120         508
                    372:     Line numbers                   0           0           0         380
                    373:     Optimization symbols           0           0           0           0
                    374:     Procedures                     1           0           0         436
                    375:     Auxiliary symbols             14          25         100         684
                    376:     Relative Files                 0           0           0           0
                    377: 
                    378:     There are 10 local symbols, starting at 442
                    379: 
                    380:        Symbol# 0: "/usr/include/stdio.h"
                    381:            End+1 symbol  = 10
                    382:            String index  = 1
                    383:            Storage class = Text        Index  = 10
                    384:            Symbol type   = File        Value  = 0
                    385: 
                    386:        Symbol# 1: "_iobuf"
                    387:            End+1 symbol  = 9
                    388:            String index  = 22
                    389:            Storage class = Info        Index  = 9
                    390:            Symbol type   = Block       Value  = 20
                    391: 
                    392:        Symbol# 2: "_cnt"
                    393:            Type          = int
                    394:            String index  = 29
                    395:            Storage class = Info        Index  = 4
                    396:            Symbol type   = Member      Value  = 0
                    397: 
                    398:        Symbol# 3: "_ptr"
                    399:            Type          = ptr to char
                    400:            String index  = 34
                    401:            Storage class = Info        Index  = 15
                    402:            Symbol type   = Member      Value  = 32
                    403: 
                    404:        Symbol# 4: "_base"
                    405:            Type          = ptr to char
                    406:            String index  = 39
                    407:            Storage class = Info        Index  = 16
                    408:            Symbol type   = Member      Value  = 64
                    409: 
                    410:        Symbol# 5: "_bufsiz"
                    411:            Type          = int
                    412:            String index  = 45
                    413:            Storage class = Info        Index  = 4
                    414:            Symbol type   = Member      Value  = 96
                    415: 
                    416:        Symbol# 6: "_flag"
                    417:            Type          = short
                    418:            String index  = 53
                    419:            Storage class = Info        Index  = 3
                    420:            Symbol type   = Member      Value  = 128
                    421: 
                    422:        Symbol# 7: "_file"
                    423:            Type          = char
                    424:            String index  = 59
                    425:            Storage class = Info        Index  = 2
                    426:            Symbol type   = Member      Value  = 144
                    427: 
                    428:        Symbol# 8: ""
                    429:            First symbol  = 1
                    430:            String index  = 0
                    431:            Storage class = Info        Index  = 1
                    432:            Symbol type   = End         Value  = 0
                    433: 
                    434:        Symbol# 9: "/usr/include/stdio.h"
                    435:            First symbol  = 0
                    436:            String index  = 1
                    437:            Storage class = Text        Index  = 0
                    438:            Symbol type   = End         Value  = 0
                    439: 
                    440:     There are 25 auxiliary table entries, starting at 642.
                    441: 
                    442:        * #14             -1, [4095/1048575], [63 1:1 f:f:f:f:f:f]
                    443:          #15          65544, [   8/     16], [ 2 0:0 1:0:0:0:0:0]
                    444:          #16          65544, [   8/     16], [ 2 0:0 1:0:0:0:0:0]
                    445:        * #17         196656, [  48/     48], [12 0:0 3:0:0:0:0:0]
                    446:        * #18           8191, [4095/      1], [63 1:1 0:0:0:0:f:1]
                    447:        * #19              1, [   1/      0], [ 0 1:0 0:0:0:0:0:0]
                    448:        * #20          20479, [4095/      4], [63 1:1 0:0:0:0:f:4]
                    449:        * #21              1, [   1/      0], [ 0 1:0 0:0:0:0:0:0]
                    450:        * #22              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    451:        * #23              2, [   2/      0], [ 0 0:1 0:0:0:0:0:0]
                    452:        * #24            160, [ 160/      0], [40 0:0 0:0:0:0:0:0]
                    453:        * #25              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    454:        * #26              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    455:        * #27              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    456:        * #28              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    457:        * #29              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    458:        * #30              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    459:        * #31              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    460:        * #32              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    461:        * #33              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    462:        * #34              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    463:        * #35              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    464:        * #36              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    465:        * #37              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    466:        * #38              0, [   0/      0], [ 0 0:0 0:0:0:0:0:0]
                    467: 
                    468:     There are 0 procedure descriptor entries, starting at 1.
                    469: 
                    470:    There are 20 external symbols, starting at 1152
                    471: 
                    472:        Symbol# 0: "_iob"
                    473:            Type          = array [3 {160}] of struct _iobuf { ifd = 1, index = 1 }
                    474:            String index  = 0           Ifd    = 1
                    475:            Storage class = Nil         Index  = 17
                    476:            Symbol type   = Global      Value  = 60
                    477: 
                    478:        Symbol# 1: "fopen"
                    479:            String index  = 5           Ifd    = 1
                    480:            Storage class = Nil         Index  = 1048575
                    481:            Symbol type   = Proc        Value  = 0
                    482: 
                    483:        Symbol# 2: "fdopen"
                    484:            String index  = 11          Ifd    = 1
                    485:            Storage class = Nil         Index  = 1048575
                    486:            Symbol type   = Proc        Value  = 0
                    487: 
                    488:        Symbol# 3: "freopen"
                    489:            String index  = 18          Ifd    = 1
                    490:            Storage class = Nil         Index  = 1048575
                    491:            Symbol type   = Proc        Value  = 0
                    492: 
                    493:        Symbol# 4: "popen"
                    494:            String index  = 26          Ifd    = 1
                    495:            Storage class = Nil         Index  = 1048575
                    496:            Symbol type   = Proc        Value  = 0
                    497: 
                    498:        Symbol# 5: "tmpfile"
                    499:            String index  = 32          Ifd    = 1
                    500:            Storage class = Nil         Index  = 1048575
                    501:            Symbol type   = Proc        Value  = 0
                    502: 
                    503:        Symbol# 6: "ftell"
                    504:            String index  = 40          Ifd    = 1
                    505:            Storage class = Nil         Index  = 1048575
                    506:            Symbol type   = Proc        Value  = 0
                    507: 
                    508:        Symbol# 7: "rewind"
                    509:            String index  = 46          Ifd    = 1
                    510:            Storage class = Nil         Index  = 1048575
                    511:            Symbol type   = Proc        Value  = 0
                    512: 
                    513:        Symbol# 8: "setbuf"
                    514:            String index  = 53          Ifd    = 1
                    515:            Storage class = Nil         Index  = 1048575
                    516:            Symbol type   = Proc        Value  = 0
                    517: 
                    518:        Symbol# 9: "setbuffer"
                    519:            String index  = 60          Ifd    = 1
                    520:            Storage class = Nil         Index  = 1048575
                    521:            Symbol type   = Proc        Value  = 0
                    522: 
                    523:        Symbol# 10: "setlinebuf"
                    524:            String index  = 70          Ifd    = 1
                    525:            Storage class = Nil         Index  = 1048575
                    526:            Symbol type   = Proc        Value  = 0
                    527: 
                    528:        Symbol# 11: "fgets"
                    529:            String index  = 81          Ifd    = 1
                    530:            Storage class = Nil         Index  = 1048575
                    531:            Symbol type   = Proc        Value  = 0
                    532: 
                    533:        Symbol# 12: "gets"
                    534:            String index  = 87          Ifd    = 1
                    535:            Storage class = Nil         Index  = 1048575
                    536:            Symbol type   = Proc        Value  = 0
                    537: 
                    538:        Symbol# 13: "ctermid"
                    539:            String index  = 92          Ifd    = 1
                    540:            Storage class = Nil         Index  = 1048575
                    541:            Symbol type   = Proc        Value  = 0
                    542: 
                    543:        Symbol# 14: "cuserid"
                    544:            String index  = 100         Ifd    = 1
                    545:            Storage class = Nil         Index  = 1048575
                    546:            Symbol type   = Proc        Value  = 0
                    547: 
                    548:        Symbol# 15: "tempnam"
                    549:            String index  = 108         Ifd    = 1
                    550:            Storage class = Nil         Index  = 1048575
                    551:            Symbol type   = Proc        Value  = 0
                    552: 
                    553:        Symbol# 16: "tmpnam"
                    554:            String index  = 116         Ifd    = 1
                    555:            Storage class = Nil         Index  = 1048575
                    556:            Symbol type   = Proc        Value  = 0
                    557: 
                    558:        Symbol# 17: "sprintf"
                    559:            String index  = 123         Ifd    = 1
                    560:            Storage class = Nil         Index  = 1048575
                    561:            Symbol type   = Proc        Value  = 0
                    562: 
                    563:        Symbol# 18: "main"
                    564:            Type          = int
                    565:            String index  = 131         Ifd    = 0
                    566:            Storage class = Text        Index  = 1
                    567:            Symbol type   = Proc        Value  = 0
                    568: 
                    569:        Symbol# 19: "printf"
                    570:            String index  = 136         Ifd    = 0
                    571:            Storage class = Undefined   Index  = 1048575
                    572:            Symbol type   = Proc        Value  = 0
                    573: 
                    574:    The following auxiliary table entries were unused:
                    575: 
                    576:     #0               0  0x00000000  void
                    577:     #2               8  0x00000008  char
                    578:     #3              16  0x00000010  short
                    579:     #4              24  0x00000018  int
                    580:     #5              32  0x00000020  long
                    581:     #6              40  0x00000028  float
                    582:     #7              44  0x0000002c  double
                    583:     #8              12  0x0000000c  unsigned char
                    584:     #9              20  0x00000014  unsigned short
                    585:     #10             28  0x0000001c  unsigned int
                    586:     #11             36  0x00000024  unsigned long
                    587:     #14              0  0x00000000  void
                    588:     #15             24  0x00000018  int
                    589:     #19             32  0x00000020  long
                    590:     #20             40  0x00000028  float
                    591:     #21             44  0x0000002c  double
                    592:     #22             12  0x0000000c  unsigned char
                    593:     #23             20  0x00000014  unsigned short
                    594:     #24             28  0x0000001c  unsigned int
                    595:     #25             36  0x00000024  unsigned long
                    596:     #26             48  0x00000030  struct no name { ifd = -1, index = 1048575 }
                    597: 
                    598: */
                    599: 
                    600: 
                    601: #include "gvarargs.h"
                    602: #include "config.h"
                    603: #include <stdio.h>
                    604: 
                    605: #ifndef __SABER__
                    606: #define saber_stop()
                    607: #endif
                    608: 
                    609: #ifndef __LINE__
                    610: #define __LINE__ 0
                    611: #endif
                    612: 
                    613: #ifdef __STDC__
                    614: typedef void *PTR_T;
                    615: typedef const void *CPTR_T;
                    616: #define __proto(x) x
                    617: #else
                    618: 
                    619: #if defined(_STDIO_H_) || defined(__STDIO_H__)         /* Ultrix 4.0, SGI */
                    620: typedef void *PTR_T;
                    621: typedef void *CPTR_T;
                    622: 
                    623: #else
                    624: typedef char *PTR_T;                                   /* Ultrix 3.1 */
                    625: typedef char *CPTR_T;
                    626: #endif
                    627: 
                    628: #define __proto(x) ()
                    629: #define const
                    630: #endif
                    631: 
                    632: /* Do to size_t being defined in sys/types.h and different
                    633:    in stddef.h, we have to do this by hand.....  Note, these
                    634:    types are correct for MIPS based systems, and may not be
                    635:    correct for other systems.  Ultrix 4.0 and Silicon Graphics
                    636:    have this fixed, but since the following is correct, and
                    637:    the fact that including stddef.h gets you GCC's version
                    638:    instead of the standard one it's not worth it to fix it.  */
                    639: 
                    640: #if defined(__OSF1__) || defined(__OSF__) || defined(__osf__)
                    641: #define Size_t         long unsigned int
                    642: #else
                    643: #define Size_t         unsigned int
                    644: #endif
                    645: #define Ptrdiff_t      long
                    646: 
                    647: /* The following might be called from obstack or malloc,
                    648:    so they can't be static.  */
                    649: 
                    650: extern void    pfatal_with_name
                    651:                                __proto((char *));
                    652: extern void    fancy_abort     __proto((void));
                    653:        void    botch           __proto((const char *));
                    654: extern PTR_T   xmalloc         __proto((Size_t));
                    655: extern PTR_T   xcalloc         __proto((Size_t, Size_t));
                    656: extern PTR_T   xrealloc        __proto((PTR_T, Size_t));
                    657: extern void    xfree           __proto((PTR_T));
                    658: 
                    659: extern void    fatal();        /* can't use prototypes here */
                    660: extern void    error();
                    661: 
                    662: 
                    663: #ifndef MIPS_DEBUGGING_INFO
                    664: 
                    665: static int      line_number;
                    666: static int      cur_line_start;
                    667: static int      debug;
                    668: static int      had_errors;
                    669: static char    *progname;
                    670: static char    *input_name;
                    671: 
                    672: int
                    673: main ()
                    674: {
                    675:   fprintf (stderr, "Mips-tfile should only be run on a MIPS computer!\n");
                    676:   exit (1);
                    677: }
                    678: 
                    679: #else                          /* MIPS_DEBUGGING defined */
                    680: 
                    681: /* The local and global symbols have a field index, so undo any defines
                    682:    of index -> strchr and rindex -> strrchr.  */
                    683: 
                    684: #undef rindex
                    685: #undef index
                    686: 
                    687: #include <sys/types.h>
                    688: #include <string.h>
                    689: #include <ctype.h>
                    690: #include <fcntl.h>
                    691: #include <errno.h>
                    692: #include <signal.h>
                    693: #include <sys/stat.h>
                    694: 
                    695: #ifndef CROSS_COMPILE
                    696: #include <a.out.h>
                    697: #else
                    698: #include "mips/a.out.h"
                    699: #endif /* CROSS_COMPILE */
                    700: 
                    701: #if defined (USG) || defined (NO_STAB_H)
                    702: #include "gstab.h"  /* If doing DBX on sysV, use our own stab.h.  */
                    703: #else
                    704: #include <stab.h>  /* On BSD, use the system's stab.h.  */
                    705: #endif /* not USG */
                    706: 
                    707: #ifdef __GNU_STAB__
                    708: #define STAB_CODE_TYPE enum __stab_debug_code
                    709: #else
                    710: #define STAB_CODE_TYPE int
                    711: #endif
                    712: 
                    713: #ifdef _OSF_SOURCE
                    714: #define HAS_STDLIB_H
                    715: #define HAS_UNISTD_H
                    716: #endif
                    717: 
                    718: #ifdef HAS_STDLIB_H
                    719: #include <stdlib.h>
                    720: #endif
                    721: 
                    722: #ifdef HAS_UNISTD_H
                    723: #include <unistd.h>
                    724: #endif
                    725: 
                    726: #ifndef errno
                    727: extern int errno;                      /* MIPS errno.h doesn't declare this */
                    728: #endif
                    729: 
                    730: #ifndef MALLOC_CHECK
                    731: #ifdef __SABER__
                    732: #define MALLOC_CHECK
                    733: #endif
                    734: #endif
                    735: 
                    736: #define IS_ASM_IDENT(ch) \
                    737:   (isalnum (ch) || (ch) == '_' || (ch) == '.' || (ch) == '$')
                    738: 
                    739: 
                    740: /* Redefinition of of storage classes as an enumeration for better
                    741:    debugging.  */
                    742: 
                    743: typedef enum sc {
                    744:   sc_Nil        = scNil,         /* no storage class */
                    745:   sc_Text       = scText,        /* text symbol */
                    746:   sc_Data       = scData,        /* initialized data symbol */
                    747:   sc_Bss        = scBss,         /* un-initialized data symbol */
                    748:   sc_Register   = scRegister,    /* value of symbol is register number */
                    749:   sc_Abs        = scAbs,         /* value of symbol is absolute */
                    750:   sc_Undefined  = scUndefined,   /* who knows? */
                    751:   sc_CdbLocal   = scCdbLocal,    /* variable's value is IN se->va.?? */
                    752:   sc_Bits       = scBits,        /* this is a bit field */
                    753:   sc_CdbSystem  = scCdbSystem,   /* value is IN CDB's address space */
                    754:   sc_RegImage   = scRegImage,    /* register value saved on stack */
                    755:   sc_Info       = scInfo,        /* symbol contains debugger information */
                    756:   sc_UserStruct         = scUserStruct,  /* addr in struct user for current process */
                    757:   sc_SData      = scSData,       /* load time only small data */
                    758:   sc_SBss       = scSBss,        /* load time only small common */
                    759:   sc_RData      = scRData,       /* load time only read only data */
                    760:   sc_Var        = scVar,         /* Var parameter (fortran,pascal) */
                    761:   sc_Common     = scCommon,      /* common variable */
                    762:   sc_SCommon    = scSCommon,     /* small common */
                    763:   sc_VarRegister = scVarRegister, /* Var parameter in a register */
                    764:   sc_Variant    = scVariant,     /* Variant record */
                    765:   sc_SUndefined         = scSUndefined,  /* small undefined(external) data */
                    766:   sc_Init       = scInit,        /* .init section symbol */
                    767:   sc_Max        = scMax          /* Max storage class+1 */
                    768: } sc_t;
                    769: 
                    770: /* Redefinition of symbol type.  */
                    771: 
                    772: typedef enum st {
                    773:   st_Nil       = stNil,        /* Nuthin' special */
                    774:   st_Global    = stGlobal,     /* external symbol */
                    775:   st_Static    = stStatic,     /* static */
                    776:   st_Param     = stParam,      /* procedure argument */
                    777:   st_Local     = stLocal,      /* local variable */
                    778:   st_Label     = stLabel,      /* label */
                    779:   st_Proc      = stProc,       /*     "      "  Procedure */
                    780:   st_Block     = stBlock,      /* beginning of block */
                    781:   st_End       = stEnd,        /* end (of anything) */
                    782:   st_Member    = stMember,     /* member (of anything  - struct/union/enum */
                    783:   st_Typedef   = stTypedef,    /* type definition */
                    784:   st_File      = stFile,       /* file name */
                    785:   st_RegReloc  = stRegReloc,   /* register relocation */
                    786:   st_Forward   = stForward,    /* forwarding address */
                    787:   st_StaticProc        = stStaticProc, /* load time only static procs */
                    788:   st_Constant  = stConstant,   /* const */
                    789:   st_Str       = stStr,        /* string */
                    790:   st_Number    = stNumber,     /* pure number (ie. 4 NOR 2+2) */
                    791:   st_Expr      = stExpr,       /* 2+2 vs. 4 */
                    792:   st_Type      = stType,       /* post-coercion SER */
                    793:   st_Max       = stMax         /* max type+1 */
                    794: } st_t;
                    795: 
                    796: /* Redefinition of type qualifiers.  */
                    797: 
                    798: typedef enum tq {
                    799:   tq_Nil       = tqNil,        /* bt is what you see */
                    800:   tq_Ptr       = tqPtr,        /* pointer */
                    801:   tq_Proc      = tqProc,       /* procedure */
                    802:   tq_Array     = tqArray,      /* duh */
                    803:   tq_Far       = tqFar,        /* longer addressing - 8086/8 land */
                    804:   tq_Vol       = tqVol,        /* volatile */
                    805:   tq_Max       = tqMax         /* Max type qualifier+1 */
                    806: } tq_t;
                    807: 
                    808: /* Redefinition of basic types.  */
                    809: 
                    810: typedef enum bt {
                    811:   bt_Nil       = btNil,        /* undefined */
                    812:   bt_Adr       = btAdr,        /* address - integer same size as pointer */
                    813:   bt_Char      = btChar,       /* character */
                    814:   bt_UChar     = btUChar,      /* unsigned character */
                    815:   bt_Short     = btShort,      /* short */
                    816:   bt_UShort    = btUShort,     /* unsigned short */
                    817:   bt_Int       = btInt,        /* int */
                    818:   bt_UInt      = btUInt,       /* unsigned int */
                    819:   bt_Long      = btLong,       /* long */
                    820:   bt_ULong     = btULong,      /* unsigned long */
                    821:   bt_Float     = btFloat,      /* float (real) */
                    822:   bt_Double    = btDouble,     /* Double (real) */
                    823:   bt_Struct    = btStruct,     /* Structure (Record) */
                    824:   bt_Union     = btUnion,      /* Union (variant) */
                    825:   bt_Enum      = btEnum,       /* Enumerated */
                    826:   bt_Typedef   = btTypedef,    /* defined via a typedef, isymRef points */
                    827:   bt_Range     = btRange,      /* subrange of int */
                    828:   bt_Set       = btSet,        /* pascal sets */
                    829:   bt_Complex   = btComplex,    /* fortran complex */
                    830:   bt_DComplex  = btDComplex,   /* fortran double complex */
                    831:   bt_Indirect  = btIndirect,   /* forward or unnamed typedef */
                    832:   bt_FixedDec  = btFixedDec,   /* Fixed Decimal */
                    833:   bt_FloatDec  = btFloatDec,   /* Float Decimal */
                    834:   bt_String    = btString,     /* Varying Length Character String */
                    835:   bt_Bit       = btBit,        /* Aligned Bit String */
                    836:   bt_Picture   = btPicture,    /* Picture */
                    837: 
                    838: #ifdef btVoid
                    839:   bt_Void      = btVoid,       /* Void */
                    840: #else
                    841: #define bt_Void        bt_Nil
                    842: #endif
                    843: 
                    844:   bt_Max       = btMax         /* Max basic type+1 */
                    845: } bt_t;
                    846: 
                    847: 
                    848: 
                    849: /* Basic COFF storage classes.  */
                    850: enum coff_storage {
                    851:   C_EFCN       = -1,
                    852:   C_NULL       = 0,
                    853:   C_AUTO       = 1,
                    854:   C_EXT                = 2,
                    855:   C_STAT       = 3,
                    856:   C_REG                = 4,
                    857:   C_EXTDEF     = 5,
                    858:   C_LABEL      = 6,
                    859:   C_ULABEL     = 7,
                    860:   C_MOS                = 8,
                    861:   C_ARG                = 9,
                    862:   C_STRTAG     = 10,
                    863:   C_MOU                = 11,
                    864:   C_UNTAG      = 12,
                    865:   C_TPDEF      = 13,
                    866:   C_USTATIC    = 14,
                    867:   C_ENTAG      = 15,
                    868:   C_MOE                = 16,
                    869:   C_REGPARM    = 17,
                    870:   C_FIELD      = 18,
                    871:   C_BLOCK      = 100,
                    872:   C_FCN                = 101,
                    873:   C_EOS                = 102,
                    874:   C_FILE       = 103,
                    875:   C_LINE       = 104,
                    876:   C_ALIAS      = 105,
                    877:   C_HIDDEN     = 106,
                    878:   C_MAX                = 107
                    879: } coff_storage_t;
                    880: 
                    881: /* Regular COFF fundamental type.  */
                    882: typedef enum coff_type {
                    883:   T_NULL       = 0,
                    884:   T_ARG                = 1,
                    885:   T_CHAR       = 2,
                    886:   T_SHORT      = 3,
                    887:   T_INT                = 4,
                    888:   T_LONG       = 5,
                    889:   T_FLOAT      = 6,
                    890:   T_DOUBLE     = 7,
                    891:   T_STRUCT     = 8,
                    892:   T_UNION      = 9,
                    893:   T_ENUM       = 10,
                    894:   T_MOE                = 11,
                    895:   T_UCHAR      = 12,
                    896:   T_USHORT     = 13,
                    897:   T_UINT       = 14,
                    898:   T_ULONG      = 15,
                    899:   T_MAX                = 16
                    900: } coff_type_t;
                    901: 
                    902: /* Regular COFF derived types.  */
                    903: typedef enum coff_dt {
                    904:   DT_NON       = 0,
                    905:   DT_PTR       = 1,
                    906:   DT_FCN       = 2,
                    907:   DT_ARY       = 3,
                    908:   DT_MAX       = 4
                    909: } coff_dt_t;
                    910: 
                    911: #define N_BTMASK       017     /* bitmask to isolate basic type */
                    912: #define N_TMASK                003     /* bitmask to isolate derived type */
                    913: #define N_BT_SHIFT     4       /* # bits to shift past basic type */
                    914: #define N_TQ_SHIFT     2       /* # bits to shift derived types */
                    915: #define        N_TQ            6       /* # of type qualifiers */
                    916: 
                    917: /* States for whether to hash type or not.  */
                    918: typedef enum hash_state {
                    919:   hash_no      = 0,            /* don't hash type */
                    920:   hash_yes     = 1,            /* ok to hash type, or use previous hash */
                    921:   hash_record  = 2             /* ok to record hash, but don't use prev. */
                    922: } hash_state_t;
                    923: 
                    924: 
                    925: /* Types of different sized allocation requests.  */
                    926: enum alloc_type {
                    927:   alloc_type_none,             /* dummy value */
                    928:   alloc_type_scope,            /* nested scopes linked list */
                    929:   alloc_type_vlinks,           /* glue linking pages in varray */
                    930:   alloc_type_shash,            /* string hash element */
                    931:   alloc_type_thash,            /* type hash element */
                    932:   alloc_type_tag,              /* struct/union/tag element */
                    933:   alloc_type_forward,          /* element to hold unknown tag */
                    934:   alloc_type_thead,            /* head of type hash list */
                    935:   alloc_type_varray,           /* general varray allocation */
                    936:   alloc_type_last              /* last+1 element for array bounds */
                    937: };
                    938: 
                    939: 
                    940: #define WORD_ALIGN(x)  (((x) + (sizeof (long) - 1)) & ~ (sizeof (long) - 1))
                    941: #define DWORD_ALIGN(x) (((x) + 7) & ~7)
                    942: 
                    943: 
                    944: /* Structures to provide n-number of virtual arrays, each of which can
                    945:    grow linearly, and which are written in the object file as sequential
                    946:    pages.  On systems with a BSD malloc that define USE_MALLOC, the
                    947:    MAX_CLUSTER_PAGES should be 1 less than a power of two, since malloc
                    948:    adds it's overhead, and rounds up to the next power of 2.  Pages are
                    949:    linked together via a linked list.
                    950: 
                    951:    If PAGE_SIZE is > 4096, the string length in the shash_t structure
                    952:    can't be represented (assuming there are strings > 4096 bytes).  */
                    953: 
                    954: #ifndef PAGE_SIZE
                    955: #define PAGE_SIZE 4096         /* size of varray pages */
                    956: #endif
                    957: 
                    958: #define PAGE_USIZE ((Size_t)PAGE_SIZE)
                    959: 
                    960: 
                    961: #ifndef MAX_CLUSTER_PAGES      /* # pages to get from system */
                    962: #ifndef USE_MALLOC             /* in one memory request */
                    963: #define MAX_CLUSTER_PAGES 64
                    964: #else
                    965: #define MAX_CLUSTER_PAGES 63
                    966: #endif
                    967: #endif
                    968: 
                    969: 
                    970: /* Linked list connecting separate page allocations.  */
                    971: typedef struct vlinks {
                    972:   struct vlinks        *prev;          /* previous set of pages */
                    973:   struct vlinks *next;         /* next set of pages */
                    974:   union  page   *datum;                /* start of page */
                    975:   unsigned long         start_index;   /* starting index # of page */
                    976: } vlinks_t;
                    977: 
                    978: 
                    979: /* Virtual array header.  */
                    980: typedef struct varray {
                    981:   vlinks_t     *first;                 /* first page link */
                    982:   vlinks_t     *last;                  /* last page link */
                    983:   unsigned long         num_allocated;         /* # objects allocated */
                    984:   unsigned short object_size;          /* size in bytes of each object */
                    985:   unsigned short objects_per_page;     /* # objects that can fit on a page */
                    986:   unsigned short objects_last_page;    /* # objects allocated on last page */
                    987: } varray_t;
                    988: 
                    989: #ifndef MALLOC_CHECK
                    990: #define OBJECTS_PER_PAGE(type) (PAGE_SIZE / sizeof (type))
                    991: #else
                    992: #define OBJECTS_PER_PAGE(type) ((sizeof (type) > 1) ? 1 : PAGE_SIZE)
                    993: #endif
                    994: 
                    995: #define INIT_VARRAY(type) {    /* macro to initialize a varray */      \
                    996:   (vlinks_t *)0,               /* first */                             \
                    997:   (vlinks_t *)0,               /* last */                              \
                    998:   0,                           /* num_allocated */                     \
                    999:   sizeof (type),               /* object_size */                       \
                   1000:   OBJECTS_PER_PAGE (type),     /* objects_per_page */                  \
                   1001:   OBJECTS_PER_PAGE (type),     /* objects_last_page */                 \
                   1002: }
                   1003: 
                   1004: /* Master type for indexes within the symbol table. */
                   1005: typedef unsigned long symint_t;
                   1006: 
                   1007: 
                   1008: /* Linked list support for nested scopes (file, block, structure, etc.).  */
                   1009: typedef struct scope {
                   1010:   struct scope *prev;          /* previous scope level */
                   1011:   struct scope *free;          /* free list pointer */
                   1012:   SYMR         *lsym;          /* pointer to local symbol node */
                   1013:   symint_t      lnumber;       /* lsym index */
                   1014:   st_t          type;          /* type of the node */
                   1015: } scope_t;
                   1016: 
                   1017: 
                   1018: /* Forward reference list for tags referenced, but not yet defined.  */
                   1019: typedef struct forward {
                   1020:   struct forward *next;                /* next forward reference */
                   1021:   struct forward *free;                /* free list pointer */
                   1022:   AUXU          *ifd_ptr;      /* pointer to store file index */
                   1023:   AUXU          *index_ptr;    /* pointer to store symbol index */
                   1024:   AUXU          *type_ptr;     /* pointer to munge type info */
                   1025: } forward_t;
                   1026: 
                   1027: 
                   1028: /* Linked list support for tags.  The first tag in the list is always
                   1029:    the current tag for that block.  */
                   1030: typedef struct tag {
                   1031:   struct tag    *free;         /* free list pointer */
                   1032:   struct shash  *hash_ptr;     /* pointer to the hash table head */
                   1033:   struct tag    *same_name;    /* tag with same name in outer scope */
                   1034:   struct tag    *same_block;   /* next tag defined in the same block.  */
                   1035:   struct forward *forward_ref; /* list of forward references */
                   1036:   bt_t           basic_type;   /* bt_Struct, bt_Union, or bt_Enum */
                   1037:   symint_t       ifd;          /* file # tag defined in */
                   1038:   symint_t       indx;         /* index within file's local symbols */
                   1039: } tag_t;
                   1040: 
                   1041: 
                   1042: /* Head of a block's linked list of tags.  */
                   1043: typedef struct thead {
                   1044:   struct thead *prev;          /* previous block */
                   1045:   struct thead *free;          /* free list pointer */
                   1046:   struct tag   *first_tag;     /* first tag in block defined */
                   1047: } thead_t;
                   1048: 
                   1049: 
                   1050: /* Union containing pointers to each the small structures which are freed up.  */
                   1051: typedef union small_free {
                   1052:   scope_t      *f_scope;       /* scope structure */
                   1053:   thead_t      *f_thead;       /* tag head structure */
                   1054:   tag_t                *f_tag;         /* tag element structure */
                   1055:   forward_t    *f_forward;     /* forward tag reference */
                   1056: } small_free_t;
                   1057: 
                   1058: 
                   1059: /* String hash table support.  The size of the hash table must fit
                   1060:    within a page.  */
                   1061: 
                   1062: #ifndef SHASH_SIZE
                   1063: #define SHASH_SIZE 1009
                   1064: #endif
                   1065: 
                   1066: #define HASH_LEN_MAX ((1 << 12) - 1)   /* Max length we can store */
                   1067: 
                   1068: typedef struct shash {
                   1069:   struct shash *next;          /* next hash value */
                   1070:   char         *string;        /* string we are hashing */
                   1071:   symint_t      len;           /* string length */
                   1072:   symint_t      indx;          /* index within string table */
                   1073:   EXTR         *esym_ptr;      /* global symbol pointer */
                   1074:   SYMR         *sym_ptr;       /* local symbol pointer */
                   1075:   SYMR         *end_ptr;       /* symbol pointer to end block */
                   1076:   tag_t                *tag_ptr;       /* tag pointer */
                   1077:   PDR          *proc_ptr;      /* procedure descriptor pointer */
                   1078: } shash_t;
                   1079: 
                   1080: 
                   1081: /* Type hash table support.  The size of the hash table must fit
                   1082:    within a page with the other extended file descriptor information.
                   1083:    Because unique types which are hashed are fewer in number than
                   1084:    strings, we use a smaller hash value.  */
                   1085: 
                   1086: #ifndef THASH_SIZE
                   1087: #define THASH_SIZE 113
                   1088: #endif
                   1089: 
                   1090: typedef struct thash {
                   1091:   struct thash *next;          /* next hash value */
                   1092:   AUXU          type;          /* type we are hashing */
                   1093:   symint_t      indx;          /* index within string table */
                   1094: } thash_t;
                   1095: 
                   1096: 
                   1097: /* Extended file descriptor that contains all of the support necessary
                   1098:    to add things to each file separately.  */
                   1099: typedef struct efdr {
                   1100:   FDR           fdr;           /* File header to be written out */
                   1101:   FDR          *orig_fdr;      /* original file header */
                   1102:   char         *name;          /* filename */
                   1103:   int           name_len;      /* length of the filename */
                   1104:   symint_t      void_type;     /* aux. pointer to 'void' type */
                   1105:   symint_t      int_type;      /* aux. pointer to 'int' type */
                   1106:   scope_t      *cur_scope;     /* current nested scopes */
                   1107:   symint_t      file_index;    /* current file number */
                   1108:   int           nested_scopes; /* # nested scopes */
                   1109:   varray_t      strings;       /* local strings */
                   1110:   varray_t      symbols;       /* local symbols */
                   1111:   varray_t      procs;         /* procedures */
                   1112:   varray_t      aux_syms;      /* auxiliary symbols */
                   1113:   struct efdr  *next_file;     /* next file descriptor */
                   1114:                                /* string/type hash tables */
                   1115:   shash_t      **shash_head;   /* string hash table */
                   1116:   thash_t      *thash_head[THASH_SIZE];
                   1117: } efdr_t;
                   1118: 
                   1119: /* Pre-initialized extended file structure.  */
                   1120: static efdr_t init_file = 
                   1121: {
                   1122:   {                    /* FDR structure */
                   1123:     0,                 /* adr:         memory address of beginning of file */
                   1124:     0,                 /* rss:         file name (of source, if known) */
                   1125:     0,                 /* issBase:     file's string space */
                   1126:     0,                 /* cbSs:        number of bytes in the ss */
                   1127:     0,                 /* isymBase:    beginning of symbols */
                   1128:     0,                 /* csym:        count file's of symbols */
                   1129:     0,                 /* ilineBase:   file's line symbols */
                   1130:     0,                 /* cline:       count of file's line symbols */
                   1131:     0,                 /* ioptBase:    file's optimization entries */
                   1132:     0,                 /* copt:        count of file's optimization entries */
                   1133:     0,                 /* ipdFirst:    start of procedures for this file */
                   1134:     0,                 /* cpd:         count of procedures for this file */
                   1135:     0,                 /* iauxBase:    file's auxiliary entries */
                   1136:     0,                 /* caux:        count of file's auxiliary entries */
                   1137:     0,                 /* rfdBase:     index into the file indirect table */
                   1138:     0,                 /* crfd:        count file indirect entries */
                   1139:     langC,             /* lang:        language for this file */
                   1140:     1,                 /* fMerge:      whether this file can be merged */
                   1141:     0,                 /* fReadin:     true if read in (not just created) */
                   1142: #if BYTES_BIG_ENDIAN
                   1143:     1,                 /* fBigendian:  if 1, compiled on big endian machine */
                   1144: #else
                   1145:     0,                 /* fBigendian:  if 1, compiled on big endian machine */
                   1146: #endif
                   1147:     GLEVEL_2,          /* glevel:      level this file was compiled with */
                   1148:     0,                 /* reserved:    reserved for future use */
                   1149:     0,                 /* cbLineOffset: byte offset from header for this file ln's */
                   1150:     0,                 /* cbLine:      size of lines for this file */
                   1151:   },
                   1152: 
                   1153:   (FDR *)0,            /* orig_fdr:    original file header pointer */
                   1154:   (char *)0,           /* name:        pointer to filename */
                   1155:   0,                   /* name_len:    length of filename */
                   1156:   0,                   /* void_type:   ptr to aux node for void type */
                   1157:   0,                   /* int_type:    ptr to aux node for int type */
                   1158:   (scope_t *)0,                /* cur_scope:   current scope being processed */
                   1159:   0,                   /* file_index:  current file # */
                   1160:   0,                   /* nested_scopes: # nested scopes */
                   1161:   INIT_VARRAY (char),  /* strings:     local string varray */
                   1162:   INIT_VARRAY (SYMR),  /* symbols:     local symbols varray */
                   1163:   INIT_VARRAY (PDR),   /* procs:       procedure varray */
                   1164:   INIT_VARRAY (AUXU),  /* aux_syms:    auxiliary symbols varray */
                   1165: 
                   1166:   (struct efdr *)0,    /* next_file:   next file structure */
                   1167: 
                   1168:   (shash_t **)0,       /* shash_head:  string hash table */
                   1169:   { 0 },               /* thash_head:  type hash table */
                   1170: };
                   1171: 
                   1172: 
                   1173: static efdr_t *first_file;                     /* first file descriptor */
                   1174: static efdr_t **last_file_ptr = &first_file;   /* file descriptor tail */
                   1175: 
                   1176: 
                   1177: /* Union of various things that are held in pages.  */
                   1178: typedef union page {
                   1179:   char         byte    [ PAGE_SIZE ];
                   1180:   unsigned char        ubyte   [ PAGE_SIZE ];
                   1181:   efdr_t       file    [ PAGE_SIZE / sizeof (efdr_t)    ];
                   1182:   FDR          ofile   [ PAGE_SIZE / sizeof (FDR)       ];
                   1183:   PDR          proc    [ PAGE_SIZE / sizeof (PDR)       ];
                   1184:   SYMR         sym     [ PAGE_SIZE / sizeof (SYMR)      ];
                   1185:   EXTR         esym    [ PAGE_SIZE / sizeof (EXTR)      ];
                   1186:   AUXU         aux     [ PAGE_SIZE / sizeof (AUXU)      ];
                   1187:   DNR          dense   [ PAGE_SIZE / sizeof (DNR)       ];
                   1188:   scope_t      scope   [ PAGE_SIZE / sizeof (scope_t)   ];
                   1189:   vlinks_t     vlinks  [ PAGE_SIZE / sizeof (vlinks_t)  ];
                   1190:   shash_t      shash   [ PAGE_SIZE / sizeof (shash_t)   ];
                   1191:   thash_t      thash   [ PAGE_SIZE / sizeof (thash_t)   ];
                   1192:   tag_t                tag     [ PAGE_SIZE / sizeof (tag_t)     ];
                   1193:   forward_t    forward [ PAGE_SIZE / sizeof (forward_t) ];
                   1194:   thead_t      thead   [ PAGE_SIZE / sizeof (thead_t)   ];
                   1195: } page_t;
                   1196: 
                   1197: 
                   1198: /* Structure holding allocation information for small sized structures.  */
                   1199: typedef struct alloc_info {
                   1200:   char         *alloc_name;    /* name of this allocation type (must be first) */
                   1201:   page_t       *cur_page;      /* current page being allocated from */
                   1202:   small_free_t  free_list;     /* current free list if any */
                   1203:   int           unallocated;   /* number of elements unallocated on page */
                   1204:   int           total_alloc;   /* total number of allocations */
                   1205:   int           total_free;    /* total number of frees */
                   1206:   int           total_pages;   /* total number of pages allocated */
                   1207: } alloc_info_t;
                   1208: 
                   1209: /* Type information collected together.  */
                   1210: typedef struct type_info {
                   1211:   bt_t       basic_type;               /* basic type */
                   1212:   coff_type_t orig_type;               /* original COFF-based type */
                   1213:   int        num_tq;                   /* # type qualifiers */
                   1214:   int        num_dims;                 /* # dimensions */
                   1215:   int        num_sizes;                /* # sizes */
                   1216:   int        extra_sizes;              /* # extra sizes not tied with dims */
                   1217:   tag_t *     tag_ptr;                 /* tag pointer */
                   1218:   int        bitfield;                 /* symbol is a bitfield */
                   1219:   int        unknown_tag;              /* this is an unknown tag */
                   1220:   tq_t       type_qualifiers[N_TQ];    /* type qualifiers (ptr, func, array)*/
                   1221:   symint_t    dimensions     [N_TQ];   /* dimensions for each array */
                   1222:   symint_t    sizes         [N_TQ+2];  /* sizes of each array slice + size of
                   1223:                                           struct/union/enum + bitfield size */
                   1224: } type_info_t;
                   1225: 
                   1226: /* Pre-initialized type_info struct.  */
                   1227: static type_info_t type_info_init = {
                   1228:   bt_Nil,                              /* basic type */
                   1229:   T_NULL,                              /* original COFF-based type */
                   1230:   0,                                   /* # type qualifiers */
                   1231:   0,                                   /* # dimensions */
                   1232:   0,                                   /* # sizes */
                   1233:   0,                                   /* sizes not tied with dims */
                   1234:   NULL,                                        /* ptr to tag */
                   1235:   0,                                   /* bitfield */
                   1236:   0,                                   /* unknown tag */
                   1237:   {                                    /* type qualifiers */
                   1238:     tq_Nil,
                   1239:     tq_Nil,
                   1240:     tq_Nil,
                   1241:     tq_Nil,
                   1242:     tq_Nil,
                   1243:     tq_Nil,
                   1244:   },
                   1245:   {                                    /* dimensions */
                   1246:     0,
                   1247:     0,
                   1248:     0,
                   1249:     0,
                   1250:     0,
                   1251:     0
                   1252:   },
                   1253:   {                                    /* sizes */
                   1254:     0,
                   1255:     0,
                   1256:     0,
                   1257:     0,
                   1258:     0,
                   1259:     0,
                   1260:     0,
                   1261:     0,
                   1262:   },
                   1263: };
                   1264: 
                   1265: 
                   1266: /* Global virtual arrays & hash table for external strings as well as
                   1267:    for the tags table and global tables for file descriptors, and
                   1268:    dense numbers.  */
                   1269: 
                   1270: static varray_t file_desc      = INIT_VARRAY (efdr_t);
                   1271: static varray_t dense_num      = INIT_VARRAY (DNR);
                   1272: static varray_t tag_strings    = INIT_VARRAY (char);
                   1273: static varray_t ext_strings    = INIT_VARRAY (char);
                   1274: static varray_t ext_symbols    = INIT_VARRAY (EXTR);
                   1275: 
                   1276: static shash_t *orig_str_hash[SHASH_SIZE];
                   1277: static shash_t *ext_str_hash [SHASH_SIZE];
                   1278: static shash_t *tag_hash     [SHASH_SIZE];
                   1279: 
                   1280: /* Static types for int and void.  Also, remember the last function's
                   1281:    type (which is set up when we encounter the declaration for the
                   1282:    function, and used when the end block for the function is emitted.  */
                   1283: 
                   1284: static type_info_t int_type_info;
                   1285: static type_info_t void_type_info;
                   1286: static type_info_t last_func_type_info;
                   1287: static EXTR      *last_func_eptr;
                   1288: 
                   1289: 
                   1290: /* Convert COFF basic type to ECOFF basic type.  The T_NULL type
                   1291:    really should use bt_Void, but this causes the current ecoff GDB to
                   1292:    issue unsupported type messages, and the Ultrix 4.00 dbx (aka MIPS
                   1293:    2.0) doesn't understand it, even though the compiler generates it.
                   1294:    Maybe this will be fixed in 2.10 or 2.20 of the MIPS compiler
                   1295:    suite, but for now go with what works.  */
                   1296: 
                   1297: static bt_t map_coff_types[ (int)T_MAX ] = {
                   1298:   bt_Nil,                      /* T_NULL */
                   1299:   bt_Nil,                      /* T_ARG */
                   1300:   bt_Char,                     /* T_CHAR */
                   1301:   bt_Short,                    /* T_SHORT */
                   1302:   bt_Int,                      /* T_INT */
                   1303:   bt_Long,                     /* T_LONG */
                   1304:   bt_Float,                    /* T_FLOAT */
                   1305:   bt_Double,                   /* T_DOUBLE */
                   1306:   bt_Struct,                   /* T_STRUCT */
                   1307:   bt_Union,                    /* T_UNION */
                   1308:   bt_Enum,                     /* T_ENUM */
                   1309:   bt_Enum,                     /* T_MOE */
                   1310:   bt_UChar,                    /* T_UCHAR */
                   1311:   bt_UShort,                   /* T_USHORT */
                   1312:   bt_UInt,                     /* T_UINT */
                   1313:   bt_ULong                     /* T_ULONG */
                   1314: };
                   1315: 
                   1316: /* Convert COFF storage class to ECOFF storage class.  */
                   1317: static sc_t map_coff_storage[ (int)C_MAX ] = {
                   1318:   sc_Nil,                      /*   0: C_NULL */
                   1319:   sc_Abs,                      /*   1: C_AUTO    auto var */
                   1320:   sc_Undefined,                        /*   2: C_EXT     external */
                   1321:   sc_Data,                     /*   3: C_STAT    static */
                   1322:   sc_Register,                 /*   4: C_REG     register */
                   1323:   sc_Undefined,                        /*   5: C_EXTDEF  ??? */
                   1324:   sc_Text,                     /*   6: C_LABEL   label */
                   1325:   sc_Text,                     /*   7: C_ULABEL  user label */
                   1326:   sc_Info,                     /*   8: C_MOS     member of struct */
                   1327:   sc_Abs,                      /*   9: C_ARG     argument */
                   1328:   sc_Info,                     /*  10: C_STRTAG  struct tag */
                   1329:   sc_Info,                     /*  11: C_MOU     member of union */
                   1330:   sc_Info,                     /*  12: C_UNTAG   union tag */
                   1331:   sc_Info,                     /*  13: C_TPDEF   typedef */
                   1332:   sc_Data,                     /*  14: C_USTATIC ??? */
                   1333:   sc_Info,                     /*  15: C_ENTAG   enum tag */
                   1334:   sc_Info,                     /*  16: C_MOE     member of enum */
                   1335:   sc_Register,                 /*  17: C_REGPARM register parameter */
                   1336:   sc_Bits,                     /*  18; C_FIELD   bitfield */
                   1337:   sc_Nil,                      /*  19 */
                   1338:   sc_Nil,                      /*  20 */
                   1339:   sc_Nil,                      /*  21 */
                   1340:   sc_Nil,                      /*  22 */
                   1341:   sc_Nil,                      /*  23 */
                   1342:   sc_Nil,                      /*  24 */
                   1343:   sc_Nil,                      /*  25 */
                   1344:   sc_Nil,                      /*  26 */
                   1345:   sc_Nil,                      /*  27 */
                   1346:   sc_Nil,                      /*  28 */
                   1347:   sc_Nil,                      /*  29 */
                   1348:   sc_Nil,                      /*  30 */
                   1349:   sc_Nil,                      /*  31 */
                   1350:   sc_Nil,                      /*  32 */
                   1351:   sc_Nil,                      /*  33 */
                   1352:   sc_Nil,                      /*  34 */
                   1353:   sc_Nil,                      /*  35 */
                   1354:   sc_Nil,                      /*  36 */
                   1355:   sc_Nil,                      /*  37 */
                   1356:   sc_Nil,                      /*  38 */
                   1357:   sc_Nil,                      /*  39 */
                   1358:   sc_Nil,                      /*  40 */
                   1359:   sc_Nil,                      /*  41 */
                   1360:   sc_Nil,                      /*  42 */
                   1361:   sc_Nil,                      /*  43 */
                   1362:   sc_Nil,                      /*  44 */
                   1363:   sc_Nil,                      /*  45 */
                   1364:   sc_Nil,                      /*  46 */
                   1365:   sc_Nil,                      /*  47 */
                   1366:   sc_Nil,                      /*  48 */
                   1367:   sc_Nil,                      /*  49 */
                   1368:   sc_Nil,                      /*  50 */
                   1369:   sc_Nil,                      /*  51 */
                   1370:   sc_Nil,                      /*  52 */
                   1371:   sc_Nil,                      /*  53 */
                   1372:   sc_Nil,                      /*  54 */
                   1373:   sc_Nil,                      /*  55 */
                   1374:   sc_Nil,                      /*  56 */
                   1375:   sc_Nil,                      /*  57 */
                   1376:   sc_Nil,                      /*  58 */
                   1377:   sc_Nil,                      /*  59 */
                   1378:   sc_Nil,                      /*  60 */
                   1379:   sc_Nil,                      /*  61 */
                   1380:   sc_Nil,                      /*  62 */
                   1381:   sc_Nil,                      /*  63 */
                   1382:   sc_Nil,                      /*  64 */
                   1383:   sc_Nil,                      /*  65 */
                   1384:   sc_Nil,                      /*  66 */
                   1385:   sc_Nil,                      /*  67 */
                   1386:   sc_Nil,                      /*  68 */
                   1387:   sc_Nil,                      /*  69 */
                   1388:   sc_Nil,                      /*  70 */
                   1389:   sc_Nil,                      /*  71 */
                   1390:   sc_Nil,                      /*  72 */
                   1391:   sc_Nil,                      /*  73 */
                   1392:   sc_Nil,                      /*  74 */
                   1393:   sc_Nil,                      /*  75 */
                   1394:   sc_Nil,                      /*  76 */
                   1395:   sc_Nil,                      /*  77 */
                   1396:   sc_Nil,                      /*  78 */
                   1397:   sc_Nil,                      /*  79 */
                   1398:   sc_Nil,                      /*  80 */
                   1399:   sc_Nil,                      /*  81 */
                   1400:   sc_Nil,                      /*  82 */
                   1401:   sc_Nil,                      /*  83 */
                   1402:   sc_Nil,                      /*  84 */
                   1403:   sc_Nil,                      /*  85 */
                   1404:   sc_Nil,                      /*  86 */
                   1405:   sc_Nil,                      /*  87 */
                   1406:   sc_Nil,                      /*  88 */
                   1407:   sc_Nil,                      /*  89 */
                   1408:   sc_Nil,                      /*  90 */
                   1409:   sc_Nil,                      /*  91 */
                   1410:   sc_Nil,                      /*  92 */
                   1411:   sc_Nil,                      /*  93 */
                   1412:   sc_Nil,                      /*  94 */
                   1413:   sc_Nil,                      /*  95 */
                   1414:   sc_Nil,                      /*  96 */
                   1415:   sc_Nil,                      /*  97 */
                   1416:   sc_Nil,                      /*  98 */
                   1417:   sc_Nil,                      /*  99 */
                   1418:   sc_Text,                     /* 100: C_BLOCK  block start/end */
                   1419:   sc_Text,                     /* 101: C_FCN    function start/end */
                   1420:   sc_Info,                     /* 102: C_EOS    end of struct/union/enum */
                   1421:   sc_Nil,                      /* 103: C_FILE   file start */
                   1422:   sc_Nil,                      /* 104: C_LINE   line number */
                   1423:   sc_Nil,                      /* 105: C_ALIAS  combined type info */
                   1424:   sc_Nil,                      /* 106: C_HIDDEN ??? */
                   1425: };
                   1426: 
                   1427: /* Convert COFF storage class to ECOFF symbol type.  */
                   1428: static st_t map_coff_sym_type[ (int)C_MAX ] = {
                   1429:   st_Nil,                      /*   0: C_NULL */
                   1430:   st_Local,                    /*   1: C_AUTO    auto var */
                   1431:   st_Global,                   /*   2: C_EXT     external */
                   1432:   st_Static,                   /*   3: C_STAT    static */
                   1433:   st_Local,                    /*   4: C_REG     register */
                   1434:   st_Global,                   /*   5: C_EXTDEF  ??? */
                   1435:   st_Label,                    /*   6: C_LABEL   label */
                   1436:   st_Label,                    /*   7: C_ULABEL  user label */
                   1437:   st_Member,                   /*   8: C_MOS     member of struct */
                   1438:   st_Param,                    /*   9: C_ARG     argument */
                   1439:   st_Block,                    /*  10: C_STRTAG  struct tag */
                   1440:   st_Member,                   /*  11: C_MOU     member of union */
                   1441:   st_Block,                    /*  12: C_UNTAG   union tag */
                   1442:   st_Typedef,                  /*  13: C_TPDEF   typedef */
                   1443:   st_Static,                   /*  14: C_USTATIC ??? */
                   1444:   st_Block,                    /*  15: C_ENTAG   enum tag */
                   1445:   st_Member,                   /*  16: C_MOE     member of enum */
                   1446:   st_Param,                    /*  17: C_REGPARM register parameter */
                   1447:   st_Member,                   /*  18; C_FIELD   bitfield */
                   1448:   st_Nil,                      /*  19 */
                   1449:   st_Nil,                      /*  20 */
                   1450:   st_Nil,                      /*  21 */
                   1451:   st_Nil,                      /*  22 */
                   1452:   st_Nil,                      /*  23 */
                   1453:   st_Nil,                      /*  24 */
                   1454:   st_Nil,                      /*  25 */
                   1455:   st_Nil,                      /*  26 */
                   1456:   st_Nil,                      /*  27 */
                   1457:   st_Nil,                      /*  28 */
                   1458:   st_Nil,                      /*  29 */
                   1459:   st_Nil,                      /*  30 */
                   1460:   st_Nil,                      /*  31 */
                   1461:   st_Nil,                      /*  32 */
                   1462:   st_Nil,                      /*  33 */
                   1463:   st_Nil,                      /*  34 */
                   1464:   st_Nil,                      /*  35 */
                   1465:   st_Nil,                      /*  36 */
                   1466:   st_Nil,                      /*  37 */
                   1467:   st_Nil,                      /*  38 */
                   1468:   st_Nil,                      /*  39 */
                   1469:   st_Nil,                      /*  40 */
                   1470:   st_Nil,                      /*  41 */
                   1471:   st_Nil,                      /*  42 */
                   1472:   st_Nil,                      /*  43 */
                   1473:   st_Nil,                      /*  44 */
                   1474:   st_Nil,                      /*  45 */
                   1475:   st_Nil,                      /*  46 */
                   1476:   st_Nil,                      /*  47 */
                   1477:   st_Nil,                      /*  48 */
                   1478:   st_Nil,                      /*  49 */
                   1479:   st_Nil,                      /*  50 */
                   1480:   st_Nil,                      /*  51 */
                   1481:   st_Nil,                      /*  52 */
                   1482:   st_Nil,                      /*  53 */
                   1483:   st_Nil,                      /*  54 */
                   1484:   st_Nil,                      /*  55 */
                   1485:   st_Nil,                      /*  56 */
                   1486:   st_Nil,                      /*  57 */
                   1487:   st_Nil,                      /*  58 */
                   1488:   st_Nil,                      /*  59 */
                   1489:   st_Nil,                      /*  60 */
                   1490:   st_Nil,                      /*  61 */
                   1491:   st_Nil,                      /*  62 */
                   1492:   st_Nil,                      /*  63 */
                   1493:   st_Nil,                      /*  64 */
                   1494:   st_Nil,                      /*  65 */
                   1495:   st_Nil,                      /*  66 */
                   1496:   st_Nil,                      /*  67 */
                   1497:   st_Nil,                      /*  68 */
                   1498:   st_Nil,                      /*  69 */
                   1499:   st_Nil,                      /*  70 */
                   1500:   st_Nil,                      /*  71 */
                   1501:   st_Nil,                      /*  72 */
                   1502:   st_Nil,                      /*  73 */
                   1503:   st_Nil,                      /*  74 */
                   1504:   st_Nil,                      /*  75 */
                   1505:   st_Nil,                      /*  76 */
                   1506:   st_Nil,                      /*  77 */
                   1507:   st_Nil,                      /*  78 */
                   1508:   st_Nil,                      /*  79 */
                   1509:   st_Nil,                      /*  80 */
                   1510:   st_Nil,                      /*  81 */
                   1511:   st_Nil,                      /*  82 */
                   1512:   st_Nil,                      /*  83 */
                   1513:   st_Nil,                      /*  84 */
                   1514:   st_Nil,                      /*  85 */
                   1515:   st_Nil,                      /*  86 */
                   1516:   st_Nil,                      /*  87 */
                   1517:   st_Nil,                      /*  88 */
                   1518:   st_Nil,                      /*  89 */
                   1519:   st_Nil,                      /*  90 */
                   1520:   st_Nil,                      /*  91 */
                   1521:   st_Nil,                      /*  92 */
                   1522:   st_Nil,                      /*  93 */
                   1523:   st_Nil,                      /*  94 */
                   1524:   st_Nil,                      /*  95 */
                   1525:   st_Nil,                      /*  96 */
                   1526:   st_Nil,                      /*  97 */
                   1527:   st_Nil,                      /*  98 */
                   1528:   st_Nil,                      /*  99 */
                   1529:   st_Block,                    /* 100: C_BLOCK  block start/end */
                   1530:   st_Proc,                     /* 101: C_FCN    function start/end */
                   1531:   st_End,                      /* 102: C_EOS    end of struct/union/enum */
                   1532:   st_File,                     /* 103: C_FILE   file start */
                   1533:   st_Nil,                      /* 104: C_LINE   line number */
                   1534:   st_Nil,                      /* 105: C_ALIAS  combined type info */
                   1535:   st_Nil,                      /* 106: C_HIDDEN ??? */
                   1536: };
                   1537: 
                   1538: /* Map COFF derived types to ECOFF type qualifiers.  */
                   1539: static tq_t map_coff_derived_type[ (int)DT_MAX ] = {
                   1540:   tq_Nil,                      /* 0: DT_NON    no more qualifiers */
                   1541:   tq_Ptr,                      /* 1: DT_PTR    pointer */
                   1542:   tq_Proc,                     /* 2: DT_FCN    function */
                   1543:   tq_Array,                    /* 3: DT_ARY    array */
                   1544: };
                   1545: 
                   1546: 
                   1547: /* Keep track of different sized allocation requests.  */
                   1548: static alloc_info_t alloc_counts[ (int)alloc_type_last ];
                   1549: 
                   1550: 
                   1551: /* Pointers and such to the original symbol table that is read in.  */
                   1552: static struct filehdr orig_file_header;                /* global object file header */
                   1553: 
                   1554: static HDRR     orig_sym_hdr;                  /* symbolic header on input */
                   1555: static char    *orig_linenum;                  /* line numbers */
                   1556: static DNR     *orig_dense;                    /* dense numbers */
                   1557: static PDR     *orig_procs;                    /* procedures */
                   1558: static SYMR    *orig_local_syms;               /* local symbols */
                   1559: static OPTR    *orig_opt_syms;                 /* optimization symbols */
                   1560: static AUXU    *orig_aux_syms;                 /* auxiliary symbols */
                   1561: static char    *orig_local_strs;               /* local strings */
                   1562: static char    *orig_ext_strs;                 /* external strings */
                   1563: static FDR     *orig_files;                    /* file descriptors */
                   1564: static symint_t        *orig_rfds;                     /* relative file desc's */
                   1565: static EXTR    *orig_ext_syms;                 /* external symbols */
                   1566: 
                   1567: /* Macros to convert an index into a given object within the original
                   1568:    symbol table.  */
                   1569: #define CHECK(num,max,str) \
                   1570:   (((unsigned long)num > (unsigned long)max) ? out_of_bounds (num, max, str, __LINE__) : 0)
                   1571: 
                   1572: #define ORIG_LINENUM(indx)     (CHECK ((indx), orig_sym_hdr.cbLine,    "line#"), (indx) + orig_linenum)
                   1573: #define ORIG_DENSE(indx)       (CHECK ((indx), orig_sym_hdr.idnMax,    "dense"), (indx) + orig_dense)
                   1574: #define ORIG_PROCS(indx)       (CHECK ((indx), orig_sym_hdr.ipdMax,    "procs"), (indx) + orig_procs)
                   1575: #define ORIG_FILES(indx)       (CHECK ((indx), orig_sym_hdr.ifdMax,    "funcs"), (indx) + orig_files)
                   1576: #define ORIG_LSYMS(indx)       (CHECK ((indx), orig_sym_hdr.isymMax,   "lsyms"), (indx) + orig_local_syms)
                   1577: #define ORIG_LSTRS(indx)       (CHECK ((indx), orig_sym_hdr.issMax,    "lstrs"), (indx) + orig_local_strs)
                   1578: #define ORIG_ESYMS(indx)       (CHECK ((indx), orig_sym_hdr.iextMax,   "esyms"), (indx) + orig_ext_syms)
                   1579: #define ORIG_ESTRS(indx)       (CHECK ((indx), orig_sym_hdr.issExtMax, "estrs"), (indx) + orig_ext_strs)
                   1580: #define ORIG_OPT(indx)         (CHECK ((indx), orig_sym_hdr.ioptMax,   "opt"),   (indx) + orig_opt_syms)
                   1581: #define ORIG_AUX(indx)         (CHECK ((indx), orig_sym_hdr.iauxMax,   "aux"),   (indx) + orig_aux_syms)
                   1582: #define ORIG_RFDS(indx)                (CHECK ((indx), orig_sym_hdr.crfd,      "rfds"),  (indx) + orig_rfds)
                   1583: 
                   1584: /* Various other statics.  */
                   1585: static HDRR    symbolic_header;                /* symbolic header */
                   1586: static efdr_t  *cur_file_ptr   = (efdr_t *) 0; /* current file desc. header */
                   1587: static PDR     *cur_proc_ptr   = (PDR *) 0;    /* current procedure header */
                   1588: static SYMR    *cur_oproc_begin        = (SYMR *) 0;   /* original proc. sym begin info */
                   1589: static SYMR    *cur_oproc_end  = (SYMR *) 0;   /* original proc. sym end info */
                   1590: static PDR     *cur_oproc_ptr  = (PDR *) 0;    /* current original procedure*/
                   1591: static thead_t *cur_tag_head   = (thead_t *)0; /* current tag head */
                   1592: static long    file_offset     = 0;            /* current file offset */
                   1593: static long    max_file_offset = 0;            /* maximum file offset */
                   1594: static FILE    *object_stream  = (FILE *)0;    /* file desc. to output .o */
                   1595: static FILE    *obj_in_stream  = (FILE *)0;    /* file desc. to input .o */
                   1596: static char    *progname       = (char *)0;    /* program name for errors */
                   1597: static char    *input_name     = "stdin";      /* name of input file */
                   1598: static char    *object_name    = (char *)0;    /* tmp. name of object file */
                   1599: static char    *obj_in_name    = (char *)0;    /* name of input object file */
                   1600: static char    *cur_line_start = (char *)0;    /* current line read in */
                   1601: static char    *cur_line_ptr   = (char *)0;    /* ptr within current line */
                   1602: static unsigned        cur_line_nbytes = 0;            /* # bytes for current line */
                   1603: static unsigned        cur_line_alloc  = 0;            /* # bytes total in buffer */
                   1604: static long    line_number     = 0;            /* current input line number */
                   1605: static int     debug           = 0;            /* trace functions */
                   1606: static int     version         = 0;            /* print version # */
                   1607: static int     had_errors      = 0;            /* != 0 if errors were found */
                   1608: static int     rename_output   = 0;            /* != 0 if rename output file*/
                   1609: static int     delete_input    = 0;            /* != 0 if delete input after done */
                   1610: static int     stabs_seen      = 0;            /* != 0 if stabs have been seen */
                   1611: 
                   1612: 
                   1613: /* Pseudo symbol to use when putting stabs into the symbol table.  */
                   1614: #ifndef STABS_SYMBOL
                   1615: #define STABS_SYMBOL "@stabs"
                   1616: #endif
                   1617: 
                   1618: static char stabs_symbol[] = STABS_SYMBOL;
                   1619: 
                   1620: 
                   1621: /* Forward reference for functions.  See the definition for more details.  */
                   1622: 
                   1623: #ifndef STATIC
                   1624: #define STATIC static
                   1625: #endif
                   1626: 
                   1627: STATIC int     out_of_bounds   __proto((symint_t, symint_t, const char *, int));
                   1628: 
                   1629: STATIC shash_t *hash_string    __proto((const char *,
                   1630:                                         Ptrdiff_t,
                   1631:                                         shash_t **,
                   1632:                                         symint_t *));
                   1633: 
                   1634: STATIC symint_t        add_string      __proto((varray_t *,
                   1635:                                         shash_t **,
                   1636:                                         const char *,
                   1637:                                         const char *,
                   1638:                                         shash_t **));
                   1639: 
                   1640: STATIC symint_t        add_local_symbol
                   1641:                                __proto((const char *,
                   1642:                                         const char *,
                   1643:                                         st_t,
                   1644:                                         sc_t,
                   1645:                                         symint_t,
                   1646:                                         symint_t));
                   1647: 
                   1648: STATIC symint_t        add_ext_symbol  __proto((const char *,
                   1649:                                         const char *,
                   1650:                                         st_t,
                   1651:                                         sc_t,
                   1652:                                         long,
                   1653:                                         symint_t,
                   1654:                                         int));
                   1655: 
                   1656: STATIC symint_t        add_aux_sym_symint
                   1657:                                __proto((symint_t));
                   1658: 
                   1659: STATIC symint_t        add_aux_sym_rndx
                   1660:                                __proto((int, symint_t));
                   1661: 
                   1662: STATIC symint_t        add_aux_sym_tir __proto((type_info_t *,
                   1663:                                         hash_state_t,
                   1664:                                         thash_t **));
                   1665: 
                   1666: STATIC tag_t * get_tag         __proto((const char *,
                   1667:                                         const char *,
                   1668:                                         symint_t,
                   1669:                                         bt_t));
                   1670: 
                   1671: STATIC void    add_unknown_tag __proto((tag_t *));
                   1672: 
                   1673: STATIC void    add_procedure   __proto((const char *,
                   1674:                                         const char *));
                   1675: 
                   1676: STATIC void    add_file        __proto((const char *,
                   1677:                                         const char *));
                   1678: 
                   1679: STATIC void    add_bytes       __proto((varray_t *,
                   1680:                                         char *,
                   1681:                                         Size_t));
                   1682: 
                   1683: STATIC void    add_varray_page __proto((varray_t *));
                   1684: 
                   1685: STATIC void    update_headers  __proto((void));
                   1686: 
                   1687: STATIC void    write_varray    __proto((varray_t *, off_t, const char *));
                   1688: STATIC void    write_object    __proto((void));
                   1689: STATIC char    *st_to_string   __proto((st_t));
                   1690: STATIC char    *sc_to_string   __proto((sc_t));
                   1691: STATIC char    *read_line      __proto((void));
                   1692: STATIC void    parse_input     __proto((void));
                   1693: STATIC void    mark_stabs      __proto((const char *));
                   1694: STATIC void    parse_begin     __proto((const char *));
                   1695: STATIC void    parse_bend      __proto((const char *));
                   1696: STATIC void    parse_def       __proto((const char *));
                   1697: STATIC void    parse_end       __proto((const char *));
                   1698: STATIC void    parse_ent       __proto((const char *));
                   1699: STATIC void    parse_file      __proto((const char *));
                   1700: STATIC void    parse_stabs_common
                   1701:                                __proto((const char *, const char *, const char *));
                   1702: STATIC void    parse_stabs     __proto((const char *));
                   1703: STATIC void    parse_stabn     __proto((const char *));
                   1704: STATIC page_t  *read_seek      __proto((Size_t, off_t, const char *));
                   1705: STATIC void    copy_object     __proto((void));
                   1706: 
                   1707: STATIC void    catch_signal    __proto((int));
                   1708: STATIC page_t  *allocate_page  __proto((void));
                   1709: 
                   1710: STATIC page_t  *allocate_multiple_pages
                   1711:                                __proto((Size_t));
                   1712: 
                   1713: STATIC void    free_multiple_pages
                   1714:                                __proto((page_t *, Size_t));
                   1715: 
                   1716: #ifndef MALLOC_CHECK
                   1717: STATIC page_t  *allocate_cluster
                   1718:                                __proto((Size_t));
                   1719: #endif
                   1720: 
                   1721: STATIC forward_t *allocate_forward     __proto((void));
                   1722: STATIC scope_t  *allocate_scope        __proto((void));
                   1723: STATIC shash_t  *allocate_shash        __proto((void));
                   1724: STATIC tag_t    *allocate_tag          __proto((void));
                   1725: STATIC thash_t  *allocate_thash        __proto((void));
                   1726: STATIC thead_t  *allocate_thead        __proto((void));
                   1727: STATIC vlinks_t         *allocate_vlinks       __proto((void));
                   1728: 
                   1729: STATIC void      free_forward          __proto((forward_t *));
                   1730: STATIC void      free_scope            __proto((scope_t *));
                   1731: STATIC void      free_tag              __proto((tag_t *));
                   1732: STATIC void      free_thead            __proto((thead_t *));
                   1733: 
                   1734: STATIC char     *local_index           __proto((const char *, int));
                   1735: STATIC char     *local_rindex          __proto((const char *, int));
                   1736: 
                   1737: #ifndef __alpha
                   1738: extern char  *sbrk                     __proto((int));
                   1739: extern PTR_T  malloc                   __proto((Size_t));
                   1740: extern PTR_T  calloc                   __proto((Size_t, Size_t));
                   1741: extern PTR_T  realloc                  __proto((PTR_T, Size_t));
                   1742: extern void   free                     __proto((PTR_T));
                   1743: #endif
                   1744: extern char  *mktemp                   __proto((char *));
                   1745: extern long   strtol                   __proto((const char *, char **, int));
                   1746: 
                   1747: extern char *optarg;
                   1748: extern int   optind;
                   1749: extern int   opterr;
                   1750: extern char *version_string;
                   1751: extern char *sys_siglist[NSIG + 1];
                   1752: 
                   1753: #ifndef SEEK_SET       /* Symbolic constants for the "fseek" function: */
                   1754: #define        SEEK_SET 0      /* Set file pointer to offset */
                   1755: #define        SEEK_CUR 1      /* Set file pointer to its current value plus offset */
                   1756: #define        SEEK_END 2      /* Set file pointer to the size of the file plus offset */
                   1757: #endif
                   1758: 
                   1759: 
                   1760: /* List of assembler pseudo ops and beginning sequences that need
                   1761:    special actions.  Someday, this should be a hash table, and such,
                   1762:    but for now a linear list of names and calls to memcmp will
                   1763:    do...... */
                   1764: 
                   1765: typedef struct _pseudo_ops {
                   1766:   const char *name;                    /* pseudo-op in ascii */
                   1767:   int len;                             /* length of name to compare */
                   1768:   void (*func) __proto((const char *));        /* function to handle line */
                   1769: } pseudo_ops_t;
                   1770: 
                   1771: static pseudo_ops_t pseudo_ops[] = {
                   1772:   { "#.def",   sizeof("#.def")-1,      parse_def },
                   1773:   { "#.begin", sizeof("#.begin")-1,    parse_begin },
                   1774:   { "#.bend",  sizeof("#.bend")-1,     parse_bend },
                   1775:   { ".end",    sizeof(".end")-1,       parse_end },
                   1776:   { ".ent",    sizeof(".ent")-1,       parse_ent },
                   1777:   { ".file",   sizeof(".file")-1,      parse_file },
                   1778:   { "#.stabs", sizeof("#.stabs")-1,    parse_stabs },
                   1779:   { "#.stabn", sizeof("#.stabn")-1,    parse_stabn },
                   1780:   { ".stabs",  sizeof(".stabs")-1,     parse_stabs },
                   1781:   { ".stabn",  sizeof(".stabn")-1,     parse_stabn },
                   1782:   { "#@stabs", sizeof("#@stabs")-1,    mark_stabs },
                   1783: };
                   1784: 
                   1785: 
                   1786: /* Add a page to a varray object.  */
                   1787: 
                   1788: STATIC void
                   1789: add_varray_page (vp)
                   1790:      varray_t *vp;                             /* varray to add page to */
                   1791: {
                   1792:   vlinks_t *new_links = allocate_vlinks ();
                   1793: 
                   1794: #ifdef MALLOC_CHECK
                   1795:   if (vp->object_size > 1)
                   1796:     new_links->datum = (page_t *) xcalloc (1, vp->object_size);
                   1797:   else
                   1798: #endif
                   1799:     new_links->datum = allocate_page ();
                   1800: 
                   1801:   alloc_counts[ (int)alloc_type_varray ].total_alloc++;
                   1802:   alloc_counts[ (int)alloc_type_varray ].total_pages++;
                   1803: 
                   1804:   new_links->start_index = vp->num_allocated;
                   1805:   vp->objects_last_page = 0;
                   1806: 
                   1807:   if (vp->first == (vlinks_t *)0)              /* first allocation? */
                   1808:     vp->first = vp->last = new_links;
                   1809:   else
                   1810:     {                                          /* 2nd or greater allocation */
                   1811:       new_links->prev = vp->last;
                   1812:       vp->last->next = new_links;
                   1813:       vp->last = new_links;
                   1814:     }
                   1815: }
                   1816: 
                   1817: 
                   1818: /* Compute hash code (from tree.c) */
                   1819: 
                   1820: #define HASHBITS 30
                   1821: 
                   1822: STATIC shash_t *
                   1823: hash_string (text, hash_len, hash_tbl, ret_hash_index)
                   1824:      const char *text;                 /* ptr to text to hash */
                   1825:      Ptrdiff_t hash_len;               /* length of the text */
                   1826:      shash_t **hash_tbl;               /* hash table */
                   1827:      symint_t *ret_hash_index;         /* ptr to store hash index */
                   1828: {
                   1829:   register unsigned long hi;
                   1830:   register Ptrdiff_t i;
                   1831:   register shash_t *ptr;
                   1832:   register int first_ch = *text;
                   1833: 
                   1834:   hi = hash_len;
                   1835:   for (i = 0; i < hash_len; i++)
                   1836:     hi = ((hi & 0x003fffff) * 613) + (text[i] & 0xff);
                   1837: 
                   1838:   hi &= (1 << HASHBITS) - 1;
                   1839:   hi %= SHASH_SIZE;
                   1840: 
                   1841:   if (ret_hash_index != (symint_t *)0)
                   1842:     *ret_hash_index = hi;
                   1843: 
                   1844:   for (ptr = hash_tbl[hi]; ptr != (shash_t *)0; ptr = ptr->next)
                   1845:     if (hash_len == ptr->len
                   1846:        && first_ch == ptr->string[0]
                   1847:        && memcmp ((CPTR_T) text, (CPTR_T) ptr->string, hash_len) == 0)
                   1848:       break;
                   1849: 
                   1850:   return ptr;
                   1851: }
                   1852: 
                   1853: 
                   1854: /* Add a string (and null pad) to one of the string tables.  A
                   1855:    consequence of hashing strings, is that we don't let strings
                   1856:    cross page boundaries.  The extra nulls will be ignored.  */
                   1857: 
                   1858: STATIC symint_t
                   1859: add_string (vp, hash_tbl, start, end_p1, ret_hash)
                   1860:      varray_t *vp;                     /* string virtual array */
                   1861:      shash_t **hash_tbl;               /* ptr to hash table */
                   1862:      const char *start;                        /* 1st byte in string */
                   1863:      const char *end_p1;               /* 1st byte after string */
                   1864:      shash_t **ret_hash;               /* return hash pointer */
                   1865: {
                   1866:   register Ptrdiff_t len = end_p1 - start;
                   1867:   register shash_t *hash_ptr;
                   1868:   symint_t hi;
                   1869: 
                   1870:   if (len >= PAGE_USIZE)
                   1871:     fatal ("String too big (%ld bytes)", (long) len);
                   1872: 
                   1873:   hash_ptr = hash_string (start, len, hash_tbl, &hi);
                   1874:   if (hash_ptr == (shash_t *)0)
                   1875:     {
                   1876:       register char *p;
                   1877: 
                   1878:       if (vp->objects_last_page + len >= PAGE_USIZE)
                   1879:        {
                   1880:          vp->num_allocated =
                   1881:            ((vp->num_allocated + PAGE_USIZE - 1) / PAGE_USIZE) * PAGE_USIZE;
                   1882:          add_varray_page (vp);
                   1883:        }
                   1884: 
                   1885:       hash_ptr = allocate_shash ();
                   1886:       hash_ptr->next = hash_tbl[hi];
                   1887:       hash_tbl[hi] = hash_ptr;
                   1888: 
                   1889:       hash_ptr->len = len;
                   1890:       hash_ptr->indx = vp->num_allocated;
                   1891:       hash_ptr->string = p = & vp->last->datum->byte[ vp->objects_last_page ];
                   1892: 
                   1893:       vp->objects_last_page += len+1;
                   1894:       vp->num_allocated += len+1;
                   1895: 
                   1896:       while (len-- > 0)
                   1897:        *p++ = *start++;
                   1898: 
                   1899:       *p = '\0';
                   1900:     }
                   1901: 
                   1902:   if (ret_hash != (shash_t **)0)
                   1903:     *ret_hash = hash_ptr;
                   1904: 
                   1905:   return hash_ptr->indx;
                   1906: }
                   1907: 
                   1908: 
                   1909: /* Add a local symbol.  */
                   1910: 
                   1911: STATIC symint_t
                   1912: add_local_symbol (str_start, str_end_p1, type, storage, value, indx)
                   1913:      const char *str_start;            /* first byte in string */
                   1914:      const char *str_end_p1;           /* first byte after string */
                   1915:      st_t type;                                /* symbol type */
                   1916:      sc_t storage;                     /* storage class */
                   1917:      symint_t value;                   /* value of symbol */
                   1918:      symint_t indx;                    /* index to local/aux. syms */
                   1919: {
                   1920:   register symint_t ret;
                   1921:   register SYMR *psym;
                   1922:   register scope_t *pscope;
                   1923:   register thead_t *ptag_head;
                   1924:   register tag_t *ptag;
                   1925:   register tag_t *ptag_next;
                   1926:   register varray_t *vp = &cur_file_ptr->symbols;
                   1927:   register int scope_delta = 0;
                   1928:   shash_t *hash_ptr = (shash_t *)0;
                   1929: 
                   1930:   if (vp->objects_last_page == vp->objects_per_page)
                   1931:     add_varray_page (vp);
                   1932: 
                   1933:   psym = &vp->last->datum->sym[ vp->objects_last_page++ ];
                   1934: 
                   1935:   psym->value = value;
                   1936:   psym->st = (unsigned) type;
                   1937:   psym->sc = (unsigned) storage;
                   1938:   psym->index = indx;
                   1939:   psym->iss = (str_start == (const char *)0)
                   1940:                ? 0
                   1941:                : add_string (&cur_file_ptr->strings,
                   1942:                              &cur_file_ptr->shash_head[0],
                   1943:                              str_start,
                   1944:                              str_end_p1,
                   1945:                              &hash_ptr);
                   1946: 
                   1947:   ret = vp->num_allocated++;
                   1948: 
                   1949:   if (MIPS_IS_STAB(psym))
                   1950:     return ret;
                   1951: 
                   1952:   /* Save the symbol within the hash table if this is a static
                   1953:      item, and it has a name.  */
                   1954:   if (hash_ptr != (shash_t *)0
                   1955:       && (type == st_Global || type == st_Static || type == st_Label
                   1956:          || type == st_Proc || type == st_StaticProc))
                   1957:     hash_ptr->sym_ptr = psym;
                   1958: 
                   1959:   /* push or pop a scope if appropriate.  */
                   1960:   switch (type)
                   1961:     {
                   1962:     default:
                   1963:       break;
                   1964: 
                   1965:     case st_File:                      /* beginning of file */
                   1966:     case st_Proc:                      /* procedure */
                   1967:     case st_StaticProc:                        /* static procedure */
                   1968:     case st_Block:                     /* begin scope */
                   1969:       pscope = allocate_scope ();
                   1970:       pscope->prev = cur_file_ptr->cur_scope;
                   1971:       pscope->lsym = psym;
                   1972:       pscope->lnumber = ret;
                   1973:       pscope->type = type;
                   1974:       cur_file_ptr->cur_scope = pscope;
                   1975: 
                   1976:       if (type != st_File)
                   1977:        scope_delta = 1;
                   1978: 
                   1979:       /* For every block type except file, struct, union, or
                   1980:         enumeration blocks, push a level on the tag stack.  We omit
                   1981:         file types, so that tags can span file boundaries.  */
                   1982:       if (type != st_File && storage != sc_Info)
                   1983:        {
                   1984:          ptag_head = allocate_thead ();
                   1985:          ptag_head->first_tag = 0;
                   1986:          ptag_head->prev = cur_tag_head;
                   1987:          cur_tag_head = ptag_head;
                   1988:        }
                   1989:       break;
                   1990: 
                   1991:     case st_End:
                   1992:       pscope = cur_file_ptr->cur_scope;
                   1993:       if (pscope == (scope_t *)0)
                   1994:        error ("internal error, too many st_End's");
                   1995: 
                   1996:       else
                   1997:        {
                   1998:          st_t begin_type = (st_t) pscope->lsym->st;
                   1999: 
                   2000:          if (begin_type != st_File)
                   2001:            scope_delta = -1;
                   2002: 
                   2003:          /* Except for file, structure, union, or enumeration end
                   2004:             blocks remove all tags created within this scope.  */
                   2005:          if (begin_type != st_File && storage != sc_Info)
                   2006:            {
                   2007:              ptag_head = cur_tag_head;
                   2008:              cur_tag_head = ptag_head->prev;
                   2009: 
                   2010:              for (ptag = ptag_head->first_tag;
                   2011:                   ptag != (tag_t *)0;
                   2012:                   ptag = ptag_next)
                   2013:                {
                   2014:                  if (ptag->forward_ref != (forward_t *)0)
                   2015:                    add_unknown_tag (ptag);
                   2016: 
                   2017:                  ptag_next = ptag->same_block;
                   2018:                  ptag->hash_ptr->tag_ptr = ptag->same_name;
                   2019:                  free_tag (ptag);
                   2020:                }
                   2021: 
                   2022:              free_thead (ptag_head);
                   2023:            }
                   2024: 
                   2025:          cur_file_ptr->cur_scope = pscope->prev;
                   2026:          psym->index = pscope->lnumber;        /* blk end gets begin sym # */
                   2027: 
                   2028:          if (storage != sc_Info)
                   2029:            psym->iss = pscope->lsym->iss;      /* blk end gets same name */
                   2030: 
                   2031:          if (begin_type == st_File || begin_type == st_Block)
                   2032:            pscope->lsym->index = ret+1;        /* block begin gets next sym # */
                   2033: 
                   2034:          /* Functions push two or more aux words as follows:
                   2035:             1st word: index+1 of the end symbol
                   2036:             2nd word: type of the function (plus any aux words needed).
                   2037:             Also, tie the external pointer back to the function begin symbol.  */
                   2038:          else
                   2039:            {
                   2040:              symint_t type;
                   2041:              pscope->lsym->index = add_aux_sym_symint (ret+1);
                   2042:              type = add_aux_sym_tir (&last_func_type_info,
                   2043:                                      hash_no,
                   2044:                                      &cur_file_ptr->thash_head[0]);
                   2045:              if (last_func_eptr)
                   2046:                {
                   2047:                  last_func_eptr->ifd = cur_file_ptr->file_index;
                   2048: 
                   2049:                  /* The index for an external st_Proc symbol is the index
                   2050:                     of the st_Proc symbol in the local symbol table.  */
                   2051:                  last_func_eptr->asym.index = psym->index;
                   2052:                }
                   2053:            }
                   2054: 
                   2055:          free_scope (pscope);
                   2056:        }
                   2057:     }
                   2058: 
                   2059:   cur_file_ptr->nested_scopes += scope_delta;
                   2060: 
                   2061:   if (debug && type != st_File
                   2062:       && (debug > 2 || type == st_Block || type == st_End
                   2063:          || type == st_Proc || type == st_StaticProc))
                   2064:     {
                   2065:       char *sc_str = sc_to_string (storage);
                   2066:       char *st_str = st_to_string (type);
                   2067:       int depth = cur_file_ptr->nested_scopes + (scope_delta < 0);
                   2068: 
                   2069:       fprintf (stderr,
                   2070:               "\tlsym\tv= %10ld, depth= %2d, sc= %-12s",
                   2071:               value, depth, sc_str);
                   2072: 
                   2073:       if (str_start && str_end_p1 - str_start > 0)
                   2074:        fprintf (stderr, " st= %-11s name= %.*s\n", st_str, str_end_p1 - str_start, str_start);
                   2075:       else
                   2076:        {
                   2077:          Size_t len = strlen (st_str);
                   2078:          fprintf (stderr, " st= %.*s\n", len-1, st_str);
                   2079:        }
                   2080:     }
                   2081: 
                   2082:   return ret;
                   2083: }
                   2084: 
                   2085: 
                   2086: /* Add an external symbol.  */
                   2087: 
                   2088: STATIC symint_t
                   2089: add_ext_symbol (str_start, str_end_p1, type, storage, value, indx, ifd)
                   2090:      const char *str_start;            /* first byte in string */
                   2091:      const char *str_end_p1;           /* first byte after string */
                   2092:      st_t type;                                /* symbol type */
                   2093:      sc_t storage;                     /* storage class */
                   2094:      long value;                       /* value of symbol */
                   2095:      symint_t indx;                    /* index to local/aux. syms */
                   2096:      int ifd;                          /* file index */
                   2097: {
                   2098:   register EXTR *psym;
                   2099:   register varray_t *vp = &ext_symbols;
                   2100:   shash_t *hash_ptr = (shash_t *)0;
                   2101: 
                   2102:   if (debug > 1)
                   2103:     {
                   2104:       char *sc_str = sc_to_string (storage);
                   2105:       char *st_str = st_to_string (type);
                   2106: 
                   2107:       fprintf (stderr,
                   2108:               "\tesym\tv= %10ld, ifd= %2d, sc= %-12s",
                   2109:               value, ifd, sc_str);
                   2110: 
                   2111:       if (str_start && str_end_p1 - str_start > 0)
                   2112:        fprintf (stderr, " st= %-11s name= %.*s\n", st_str, str_end_p1 - str_start, str_start);
                   2113:       else
                   2114:        fprintf (stderr, " st= %s\n", st_str);
                   2115:     }
                   2116: 
                   2117:   if (vp->objects_last_page == vp->objects_per_page)
                   2118:     add_varray_page (vp);
                   2119: 
                   2120:   psym = &vp->last->datum->esym[ vp->objects_last_page++ ];
                   2121: 
                   2122:   psym->ifd = ifd;
                   2123:   psym->asym.value = value;
                   2124:   psym->asym.st    = (unsigned) type;
                   2125:   psym->asym.sc    = (unsigned) storage;
                   2126:   psym->asym.index = indx;
                   2127:   psym->asym.iss   = (str_start == (const char *)0)
                   2128:                        ? 0
                   2129:                        : add_string (&ext_strings,
                   2130:                                      &ext_str_hash[0],
                   2131:                                      str_start,
                   2132:                                      str_end_p1,
                   2133:                                      &hash_ptr);
                   2134: 
                   2135:   hash_ptr->esym_ptr = psym;
                   2136:   return vp->num_allocated++;
                   2137: }
                   2138: 
                   2139: 
                   2140: /* Add an auxiliary symbol (passing a symint).  */
                   2141: 
                   2142: STATIC symint_t
                   2143: add_aux_sym_symint (aux_word)
                   2144:      symint_t aux_word;                /* auxiliary information word */
                   2145: {
                   2146:   register AUXU *aux_ptr;
                   2147:   register efdr_t *file_ptr = cur_file_ptr;
                   2148:   register varray_t *vp = &file_ptr->aux_syms;
                   2149: 
                   2150:   if (vp->objects_last_page == vp->objects_per_page)
                   2151:     add_varray_page (vp);
                   2152: 
                   2153:   aux_ptr = &vp->last->datum->aux[ vp->objects_last_page++ ];
                   2154:   aux_ptr->isym = aux_word;
                   2155: 
                   2156:   return vp->num_allocated++;
                   2157: }
                   2158: 
                   2159: 
                   2160: /* Add an auxiliary symbol (passing a file/symbol index combo).  */
                   2161: 
                   2162: STATIC symint_t
                   2163: add_aux_sym_rndx (file_index, sym_index)
                   2164:      int file_index;
                   2165:      symint_t sym_index;
                   2166: {
                   2167:   register AUXU *aux_ptr;
                   2168:   register efdr_t *file_ptr = cur_file_ptr;
                   2169:   register varray_t *vp = &file_ptr->aux_syms;
                   2170: 
                   2171:   if (vp->objects_last_page == vp->objects_per_page)
                   2172:     add_varray_page (vp);
                   2173: 
                   2174:   aux_ptr = &vp->last->datum->aux[ vp->objects_last_page++ ];
                   2175:   aux_ptr->rndx.rfd   = file_index;
                   2176:   aux_ptr->rndx.index = sym_index;
                   2177: 
                   2178:   return vp->num_allocated++;
                   2179: }
                   2180: 
                   2181: 
                   2182: /* Add an auxiliary symbol (passing the basic type and possibly
                   2183:    type qualifiers).  */
                   2184: 
                   2185: STATIC symint_t
                   2186: add_aux_sym_tir (t, state, hash_tbl)
                   2187:      type_info_t *t;           /* current type information */
                   2188:      hash_state_t state;       /* whether to hash type or not */
                   2189:      thash_t **hash_tbl;       /* pointer to hash table to use */
                   2190: {
                   2191:   register AUXU *aux_ptr;
                   2192:   register efdr_t *file_ptr = cur_file_ptr;
                   2193:   register varray_t *vp = &file_ptr->aux_syms;
                   2194:   static AUXU init_aux;
                   2195:   symint_t ret;
                   2196:   int i;
                   2197:   AUXU aux;
                   2198: 
                   2199:   aux = init_aux;
                   2200:   aux.ti.bt = (int) t->basic_type;
                   2201:   aux.ti.continued = 0;
                   2202:   aux.ti.fBitfield = t->bitfield;
                   2203: 
                   2204:   aux.ti.tq0 = (int) t->type_qualifiers[0];
                   2205:   aux.ti.tq1 = (int) t->type_qualifiers[1];
                   2206:   aux.ti.tq2 = (int) t->type_qualifiers[2];
                   2207:   aux.ti.tq3 = (int) t->type_qualifiers[3];
                   2208:   aux.ti.tq4 = (int) t->type_qualifiers[4];
                   2209:   aux.ti.tq5 = (int) t->type_qualifiers[5];
                   2210: 
                   2211: 
                   2212:   /* For anything that adds additional information, we must not hash,
                   2213:      so check here, and reset our state. */
                   2214: 
                   2215:   if (state != hash_no
                   2216:       && (t->type_qualifiers[0] == tq_Array
                   2217:          || t->type_qualifiers[1] == tq_Array
                   2218:          || t->type_qualifiers[2] == tq_Array
                   2219:          || t->type_qualifiers[3] == tq_Array
                   2220:          || t->type_qualifiers[4] == tq_Array
                   2221:          || t->type_qualifiers[5] == tq_Array
                   2222:          || t->basic_type == bt_Struct
                   2223:          || t->basic_type == bt_Union
                   2224:          || t->basic_type == bt_Enum
                   2225:          || t->bitfield
                   2226:          || t->num_dims > 0))
                   2227:     state = hash_no;
                   2228: 
                   2229:   /* See if we can hash this type, and save some space, but some types
                   2230:      can't be hashed (because they contain arrays or continuations),
                   2231:      and others can be put into the hash list, but cannot use existing
                   2232:      types because other aux entries precede this one.  */
                   2233: 
                   2234:   if (state != hash_no)
                   2235:     {
                   2236:       register thash_t *hash_ptr;
                   2237:       register symint_t hi;
                   2238: 
                   2239:       hi = aux.isym & ((1 << HASHBITS) - 1);
                   2240:       hi %= THASH_SIZE;
                   2241: 
                   2242:       for (hash_ptr = hash_tbl[hi];
                   2243:           hash_ptr != (thash_t *)0;
                   2244:           hash_ptr = hash_ptr->next)
                   2245:        {
                   2246:          if (aux.isym == hash_ptr->type.isym)
                   2247:            break;
                   2248:        }
                   2249: 
                   2250:       if (hash_ptr != (thash_t *)0 && state == hash_yes)
                   2251:        return hash_ptr->indx;
                   2252: 
                   2253:       if (hash_ptr == (thash_t *)0)
                   2254:        {
                   2255:          hash_ptr = allocate_thash ();
                   2256:          hash_ptr->next = hash_tbl[hi];
                   2257:          hash_ptr->type = aux;
                   2258:          hash_ptr->indx = vp->num_allocated;
                   2259:          hash_tbl[hi] = hash_ptr;
                   2260:        }
                   2261:     }
                   2262: 
                   2263:   /* Everything is set up, add the aux symbol. */
                   2264:   if (vp->objects_last_page == vp->objects_per_page)
                   2265:     add_varray_page (vp);
                   2266: 
                   2267:   aux_ptr = &vp->last->datum->aux[ vp->objects_last_page++ ];
                   2268:   *aux_ptr = aux;
                   2269: 
                   2270:   ret = vp->num_allocated++;
                   2271: 
                   2272:   /* Add bitfield length if it exists.
                   2273:      
                   2274:      NOTE:  Mips documentation claims bitfield goes at the end of the
                   2275:      AUX record, but the DECstation compiler emits it here.
                   2276:      (This would only make a difference for enum bitfields.)
                   2277: 
                   2278:      Also note:  We use the last size given since gcc may emit 2
                   2279:      for an enum bitfield.  */
                   2280: 
                   2281:   if (t->bitfield)
                   2282:     (void) add_aux_sym_symint ((symint_t)t->sizes[t->num_sizes-1]);
                   2283: 
                   2284: 
                   2285:   /* Add tag information if needed.  Structure, union, and enum
                   2286:      references add 2 aux symbols: a [file index, symbol index]
                   2287:      pointer to the structure type, and the current file index.  */
                   2288: 
                   2289:   if (t->basic_type == bt_Struct
                   2290:       || t->basic_type == bt_Union
                   2291:       || t->basic_type == bt_Enum)
                   2292:     {
                   2293:       register symint_t file_index = t->tag_ptr->ifd;
                   2294:       register symint_t sym_index  = t->tag_ptr->indx;
                   2295: 
                   2296:       if (t->unknown_tag)
                   2297:        {
                   2298:          (void) add_aux_sym_rndx (ST_RFDESCAPE, sym_index);
                   2299:          (void) add_aux_sym_symint ((symint_t)-1);
                   2300:        }
                   2301:       else if (sym_index != indexNil)
                   2302:        {
                   2303:          (void) add_aux_sym_rndx (ST_RFDESCAPE, sym_index);
                   2304:          (void) add_aux_sym_symint (file_index);
                   2305:        }
                   2306:       else
                   2307:        {
                   2308:          register forward_t *forward_ref = allocate_forward ();
                   2309: 
                   2310:          forward_ref->type_ptr = aux_ptr;
                   2311:          forward_ref->next = t->tag_ptr->forward_ref;
                   2312:          t->tag_ptr->forward_ref = forward_ref;
                   2313: 
                   2314:          (void) add_aux_sym_rndx (ST_RFDESCAPE, sym_index);
                   2315:          forward_ref->index_ptr
                   2316:            = &vp->last->datum->aux[ vp->objects_last_page - 1];
                   2317: 
                   2318:          (void) add_aux_sym_symint (file_index);
                   2319:          forward_ref->ifd_ptr
                   2320:            = &vp->last->datum->aux[ vp->objects_last_page - 1];
                   2321:        }
                   2322:     }
                   2323: 
                   2324:   /* Add information about array bounds if they exist.  */
                   2325:   for (i = 0; i < t->num_dims; i++)
                   2326:     {
                   2327:       (void) add_aux_sym_rndx (ST_RFDESCAPE,
                   2328:                               cur_file_ptr->int_type);
                   2329: 
                   2330:       (void) add_aux_sym_symint (cur_file_ptr->file_index);    /* file index*/
                   2331:       (void) add_aux_sym_symint ((symint_t)0);                 /* low bound */
                   2332:       (void) add_aux_sym_symint (t->dimensions[i] - 1);                /* high bound*/
                   2333:       (void) add_aux_sym_symint ((t->dimensions[i] == 0)       /* stride */
                   2334:                              ? 0
                   2335:                              : (t->sizes[i] * 8) / t->dimensions[i]);
                   2336:     };
                   2337: 
                   2338:   /* NOTE:  Mips documentation claism that the bitfield width goes here.
                   2339:      But it needs to be emitted earlier. */
                   2340: 
                   2341:   return ret;
                   2342: }
                   2343: 
                   2344: 
                   2345: /* Add a tag to the tag table (unless it already exists).  */
                   2346: 
                   2347: STATIC tag_t *
                   2348: get_tag (tag_start, tag_end_p1, indx, basic_type)
                   2349:      const char *tag_start;            /* 1st byte of tag name */
                   2350:      const char *tag_end_p1;           /* 1st byte after tag name */
                   2351:      symint_t indx;                    /* index of tag start block */
                   2352:      bt_t basic_type;                  /* bt_Struct, bt_Union, or bt_Enum */
                   2353: {
                   2354:   shash_t *hash_ptr;
                   2355:   tag_t *tag_ptr;
                   2356:   hash_ptr = hash_string (tag_start,
                   2357:                          tag_end_p1 - tag_start,
                   2358:                          &tag_hash[0],
                   2359:                          (symint_t *)0);
                   2360: 
                   2361:   if (hash_ptr != (shash_t *)0
                   2362:       && hash_ptr->tag_ptr != (tag_t *)0)
                   2363:   {
                   2364:     tag_ptr = hash_ptr->tag_ptr;
                   2365:     if (indx != indexNil)
                   2366:       {
                   2367:        tag_ptr->basic_type = basic_type;
                   2368:        tag_ptr->ifd        = cur_file_ptr->file_index;
                   2369:        tag_ptr->indx       = indx;
                   2370:       }
                   2371:     return tag_ptr;
                   2372:   }
                   2373: 
                   2374:   (void) add_string (&tag_strings,
                   2375:                     &tag_hash[0],
                   2376:                     tag_start,
                   2377:                     tag_end_p1,
                   2378:                     &hash_ptr);
                   2379: 
                   2380:   tag_ptr = allocate_tag ();
                   2381:   tag_ptr->forward_ref = (forward_t *) 0;
                   2382:   tag_ptr->hash_ptr    = hash_ptr;
                   2383:   tag_ptr->same_name   = hash_ptr->tag_ptr;
                   2384:   tag_ptr->basic_type  = basic_type;
                   2385:   tag_ptr->indx                = indx;
                   2386:   tag_ptr->ifd         = (indx == indexNil) ? -1 : cur_file_ptr->file_index;
                   2387:   tag_ptr->same_block  = cur_tag_head->first_tag;
                   2388: 
                   2389:   cur_tag_head->first_tag = tag_ptr;
                   2390:   hash_ptr->tag_ptr      = tag_ptr;
                   2391: 
                   2392:   return tag_ptr;
                   2393: }
                   2394: 
                   2395: 
                   2396: /* Add an unknown {struct, union, enum} tag.  */
                   2397: 
                   2398: STATIC void
                   2399: add_unknown_tag (ptag)
                   2400:      tag_t     *ptag;          /* pointer to tag information */
                   2401: {
                   2402:   shash_t *hash_ptr    = ptag->hash_ptr;
                   2403:   char *name_start     = hash_ptr->string;
                   2404:   char *name_end_p1    = name_start + hash_ptr->len;
                   2405:   forward_t *f_next    = ptag->forward_ref;
                   2406:   forward_t *f_cur;
                   2407:   int sym_index;
                   2408:   int file_index       = cur_file_ptr->file_index;
                   2409: 
                   2410:   if (debug > 1)
                   2411:     {
                   2412:       char *agg_type   = "{unknown aggregate type}";
                   2413:       switch (ptag->basic_type)
                   2414:        {
                   2415:        case bt_Struct: agg_type = "struct";    break;
                   2416:        case bt_Union:  agg_type = "union";     break;
                   2417:        case bt_Enum:   agg_type = "enum";      break;
                   2418:        default:                                break;
                   2419:        }
                   2420: 
                   2421:       fprintf (stderr, "unknown %s %.*s found\n", agg_type,
                   2422:               hash_ptr->len, name_start);
                   2423:     }
                   2424: 
                   2425:   sym_index = add_local_symbol (name_start,
                   2426:                                name_end_p1,
                   2427:                                st_Block,
                   2428:                                sc_Info,
                   2429:                                (symint_t)0,
                   2430:                                (symint_t)0);
                   2431: 
                   2432:   (void) add_local_symbol (name_start,
                   2433:                           name_end_p1,
                   2434:                           st_End,
                   2435:                           sc_Info,
                   2436:                           (symint_t)0,
                   2437:                           (symint_t)0);
                   2438: 
                   2439:   while (f_next != (forward_t *)0)
                   2440:     {
                   2441:       f_cur  = f_next;
                   2442:       f_next = f_next->next;
                   2443: 
                   2444:       f_cur->ifd_ptr->isym = file_index;
                   2445:       f_cur->index_ptr->rndx.index = sym_index;
                   2446: 
                   2447:       free_forward (f_cur);
                   2448:     }
                   2449: 
                   2450:   return;
                   2451: }
                   2452: 
                   2453: 
                   2454: /* Add a procedure to the current file's list of procedures, and record
                   2455:    this is the current procedure.  If the assembler created a PDR for
                   2456:    this procedure, use that to initialize the current PDR.  */
                   2457: 
                   2458: STATIC void
                   2459: add_procedure (func_start, func_end_p1)
                   2460:      const char *func_start;           /* 1st byte of func name */
                   2461:      const char *func_end_p1;          /* 1st byte after func name */
                   2462: {
                   2463:   register PDR *new_proc_ptr;
                   2464:   register efdr_t *file_ptr = cur_file_ptr;
                   2465:   register varray_t *vp = &file_ptr->procs;
                   2466:   register symint_t value = 0;
                   2467:   register st_t proc_type = st_Proc;
                   2468:   register shash_t *shash_ptr = hash_string (func_start,
                   2469:                                            func_end_p1 - func_start,
                   2470:                                            &orig_str_hash[0],
                   2471:                                            (symint_t *)0);
                   2472: 
                   2473:   if (debug)
                   2474:     fputc ('\n', stderr);
                   2475: 
                   2476:   if (vp->objects_last_page == vp->objects_per_page)
                   2477:     add_varray_page (vp);
                   2478: 
                   2479:   cur_proc_ptr = new_proc_ptr = &vp->last->datum->proc[ vp->objects_last_page++ ];
                   2480: 
                   2481:   vp->num_allocated++;
                   2482: 
                   2483: 
                   2484:   /* Did the assembler create this procedure?  If so, get the PDR information.  */
                   2485:   cur_oproc_ptr = (PDR *)0;
                   2486:   if (shash_ptr != (shash_t *)0)
                   2487:     {
                   2488:       register PDR *old_proc_ptr = shash_ptr->proc_ptr;
                   2489:       register SYMR *sym_ptr = shash_ptr->sym_ptr;
                   2490: 
                   2491:       if (old_proc_ptr != (PDR *)0
                   2492:          && sym_ptr != (SYMR *)0
                   2493:          && ((st_t)sym_ptr->st == st_Proc || (st_t)sym_ptr->st == st_StaticProc))
                   2494:        {
                   2495:          cur_oproc_begin = sym_ptr;
                   2496:          cur_oproc_end = shash_ptr->end_ptr;
                   2497:          value = sym_ptr->value;
                   2498: 
                   2499:          cur_oproc_ptr = old_proc_ptr;
                   2500:          proc_type = (st_t)sym_ptr->st;
                   2501:          *new_proc_ptr = *old_proc_ptr;        /* initialize */
                   2502:        }
                   2503:     }
                   2504: 
                   2505:   if (cur_oproc_ptr == (PDR *)0)
                   2506:     error ("Did not find a PDR block for %.*s", func_end_p1 - func_start, func_start);
                   2507: 
                   2508:   /* Determine the start of symbols. */
                   2509:   new_proc_ptr->isym = file_ptr->symbols.num_allocated;
                   2510: 
                   2511:   /* Push the start of the function.  */
                   2512:   (void) add_local_symbol (func_start, func_end_p1,
                   2513:                           proc_type, sc_Text,
                   2514:                           value,
                   2515:                           (symint_t)0);
                   2516: }
                   2517: 
                   2518: 
                   2519: /* Add a new filename, and set up all of the file relative
                   2520:    virtual arrays (strings, symbols, aux syms, etc.).  Record
                   2521:    where the current file structure lives.  */
                   2522: 
                   2523: STATIC void
                   2524: add_file (file_start, file_end_p1)
                   2525:      const char *file_start;           /* first byte in string */
                   2526:      const char *file_end_p1;          /* first byte after string */
                   2527: {
                   2528:   static char zero_bytes[2] = { '\0', '\0' };
                   2529: 
                   2530:   register Ptrdiff_t len = file_end_p1 - file_start;
                   2531:   register int first_ch = *file_start;
                   2532:   register efdr_t *file_ptr;
                   2533: 
                   2534:   if (debug)
                   2535:     fprintf (stderr, "\tfile\t%.*s\n", len, file_start);
                   2536: 
                   2537:   /* See if the file has already been created.  */
                   2538:   for (file_ptr = first_file;
                   2539:        file_ptr != (efdr_t *)0;
                   2540:        file_ptr = file_ptr->next_file)
                   2541:     {
                   2542:       if (first_ch == file_ptr->name[0]
                   2543:          && file_ptr->name[len] == '\0'
                   2544:          && memcmp ((CPTR_T) file_start, (CPTR_T) file_ptr->name, len) == 0)
                   2545:        {
                   2546:          cur_file_ptr = file_ptr;
                   2547:          break;
                   2548:        }
                   2549:     }
                   2550: 
                   2551:   /* If this is a new file, create it. */
                   2552:   if (file_ptr == (efdr_t *)0)
                   2553:     {
                   2554:       if (file_desc.objects_last_page == file_desc.objects_per_page)
                   2555:        add_varray_page (&file_desc);
                   2556: 
                   2557:       file_ptr = cur_file_ptr =
                   2558:        &file_desc.last->datum->file[ file_desc.objects_last_page++ ];
                   2559:       *file_ptr = init_file;
                   2560: 
                   2561:       file_ptr->file_index = file_desc.num_allocated++;
                   2562: 
                   2563:       /* Allocate the string hash table.  */
                   2564:       file_ptr->shash_head = (shash_t **) allocate_page ();
                   2565: 
                   2566:       /* Make sure 0 byte in string table is null  */
                   2567:       add_string (&file_ptr->strings,
                   2568:                  &file_ptr->shash_head[0],
                   2569:                  &zero_bytes[0],
                   2570:                  &zero_bytes[0],
                   2571:                  (shash_t **)0);
                   2572: 
                   2573:       if (file_end_p1 - file_start > PAGE_USIZE-2)
                   2574:        fatal ("Filename goes over one page boundary.");
                   2575: 
                   2576:       /* Push the start of the filename. We assume that the filename
                   2577:          will be stored at string offset 1.  */
                   2578:       (void) add_local_symbol (file_start, file_end_p1, st_File, sc_Text,
                   2579:                               (symint_t)0, (symint_t)0);
                   2580:       file_ptr->fdr.rss = 1;
                   2581:       file_ptr->name = &file_ptr->strings.last->datum->byte[1];
                   2582:       file_ptr->name_len = file_end_p1 - file_start;
                   2583: 
                   2584:       /* Update the linked list of file descriptors.  */
                   2585:       *last_file_ptr = file_ptr;
                   2586:       last_file_ptr = &file_ptr->next_file;
                   2587: 
                   2588:       /* Add void & int types to the file (void should be first to catch
                   2589:         errant 0's within the index fields).  */
                   2590:       file_ptr->void_type = add_aux_sym_tir (&void_type_info,
                   2591:                                             hash_yes,
                   2592:                                             &cur_file_ptr->thash_head[0]);
                   2593: 
                   2594:       file_ptr->int_type = add_aux_sym_tir (&int_type_info,
                   2595:                                            hash_yes,
                   2596:                                            &cur_file_ptr->thash_head[0]);
                   2597:     }
                   2598: }
                   2599: 
                   2600: 
                   2601: /* Add a stream of random bytes to a varray.  */
                   2602: 
                   2603: STATIC void
                   2604: add_bytes (vp, input_ptr, nitems)
                   2605:      varray_t *vp;                     /* virtual array to add too */
                   2606:      char *input_ptr;                  /* start of the bytes */
                   2607:      Size_t nitems;                    /* # items to move */
                   2608: {
                   2609:   register Size_t move_items;
                   2610:   register Size_t move_bytes;
                   2611:   register char *ptr;
                   2612: 
                   2613:   while (nitems > 0)
                   2614:     {
                   2615:       if (vp->objects_last_page >= vp->objects_per_page)
                   2616:        add_varray_page (vp);
                   2617: 
                   2618:       ptr = &vp->last->datum->byte[ vp->objects_last_page * vp->object_size ];
                   2619:       move_items = vp->objects_per_page - vp->objects_last_page;
                   2620:       if (move_items > nitems)
                   2621:        move_items = nitems;
                   2622: 
                   2623:       move_bytes = move_items * vp->object_size;
                   2624:       nitems -= move_items;
                   2625: 
                   2626:       if (move_bytes >= 32)
                   2627:        {
                   2628:          (void) memcpy ((PTR_T) ptr, (CPTR_T) input_ptr, move_bytes);
                   2629:          input_ptr += move_bytes;
                   2630:        }
                   2631:       else
                   2632:        {
                   2633:          while (move_bytes-- > 0)
                   2634:            *ptr++ = *input_ptr++;
                   2635:        }
                   2636:     }
                   2637: }
                   2638: 
                   2639: 
                   2640: /* Convert storage class to string.  */
                   2641: 
                   2642: STATIC char *
                   2643: sc_to_string(storage_class)
                   2644:      sc_t storage_class;
                   2645: {
                   2646:   switch(storage_class)
                   2647:     {
                   2648:     case sc_Nil:        return "Nil,";
                   2649:     case sc_Text:       return "Text,";
                   2650:     case sc_Data:       return "Data,";
                   2651:     case sc_Bss:        return "Bss,";
                   2652:     case sc_Register:   return "Register,";
                   2653:     case sc_Abs:        return "Abs,";
                   2654:     case sc_Undefined:  return "Undefined,";
                   2655:     case sc_CdbLocal:   return "CdbLocal,";
                   2656:     case sc_Bits:       return "Bits,";
                   2657:     case sc_CdbSystem:  return "CdbSystem,";
                   2658:     case sc_RegImage:   return "RegImage,";
                   2659:     case sc_Info:       return "Info,";
                   2660:     case sc_UserStruct:         return "UserStruct,";
                   2661:     case sc_SData:      return "SData,";
                   2662:     case sc_SBss:       return "SBss,";
                   2663:     case sc_RData:      return "RData,";
                   2664:     case sc_Var:        return "Var,";
                   2665:     case sc_Common:     return "Common,";
                   2666:     case sc_SCommon:    return "SCommon,";
                   2667:     case sc_VarRegister: return "VarRegister,";
                   2668:     case sc_Variant:    return "Variant,";
                   2669:     case sc_SUndefined:         return "SUndefined,";
                   2670:     case sc_Init:       return "Init,";
                   2671:     case sc_Max:        return "Max,";
                   2672:     }
                   2673: 
                   2674:   return "???,";
                   2675: }
                   2676: 
                   2677: 
                   2678: /* Convert symbol type to string.  */
                   2679: 
                   2680: STATIC char *
                   2681: st_to_string(symbol_type)
                   2682:      st_t symbol_type;
                   2683: {
                   2684:   switch(symbol_type)
                   2685:     {
                   2686:     case st_Nil:       return "Nil,";
                   2687:     case st_Global:    return "Global,";
                   2688:     case st_Static:    return "Static,";
                   2689:     case st_Param:     return "Param,";
                   2690:     case st_Local:     return "Local,";
                   2691:     case st_Label:     return "Label,";
                   2692:     case st_Proc:      return "Proc,";
                   2693:     case st_Block:     return "Block,";
                   2694:     case st_End:       return "End,";
                   2695:     case st_Member:    return "Member,";
                   2696:     case st_Typedef:   return "Typedef,";
                   2697:     case st_File:      return "File,";
                   2698:     case st_RegReloc:  return "RegReloc,";
                   2699:     case st_Forward:   return "Forward,";
                   2700:     case st_StaticProc:        return "StaticProc,";
                   2701:     case st_Constant:  return "Constant,";
                   2702:     case st_Str:       return "String,";
                   2703:     case st_Number:    return "Number,";
                   2704:     case st_Expr:      return "Expr,";
                   2705:     case st_Type:      return "Type,";
                   2706:     case st_Max:       return "Max,";
                   2707:     }
                   2708: 
                   2709:   return "???,";
                   2710: }
                   2711: 
                   2712: 
                   2713: /* Read a line from standard input, and return the start of the buffer
                   2714:    (which is grows if the line is too big).  We split lines at the
                   2715:    semi-colon, and return each logical line independently.  */
                   2716: 
                   2717: STATIC char *
                   2718: read_line __proto((void))
                   2719: {
                   2720:   static   int line_split_p    = 0;
                   2721:   register int string_p                = 0;
                   2722:   register int comment_p       = 0;
                   2723:   register int ch;
                   2724:   register char *ptr;
                   2725: 
                   2726:   if (cur_line_start == (char *)0)
                   2727:     {                          /* allocate initial page */
                   2728:       cur_line_start = (char *) allocate_page ();
                   2729:       cur_line_alloc = PAGE_SIZE;
                   2730:     }
                   2731: 
                   2732:   if (!line_split_p)
                   2733:     line_number++;
                   2734: 
                   2735:   line_split_p = 0;
                   2736:   cur_line_nbytes = 0;
                   2737: 
                   2738:   for (ptr = cur_line_start; (ch = getchar ()) != EOF; *ptr++ = ch)
                   2739:     {
                   2740:       if (++cur_line_nbytes >= cur_line_alloc-1)
                   2741:        {
                   2742:          register int num_pages = cur_line_alloc / PAGE_SIZE;
                   2743:          register char *old_buffer = cur_line_start;
                   2744: 
                   2745:          cur_line_alloc += PAGE_SIZE;
                   2746:          cur_line_start = (char *) allocate_multiple_pages (num_pages+1);
                   2747:          memcpy (cur_line_start, old_buffer, num_pages * PAGE_SIZE);
                   2748: 
                   2749:          ptr = cur_line_start + cur_line_nbytes - 1;
                   2750:        }
                   2751: 
                   2752:       if (ch == '\n')
                   2753:        {
                   2754:          *ptr++ = '\n';
                   2755:          *ptr = '\0';
                   2756:          cur_line_ptr = cur_line_start;
                   2757:          return cur_line_ptr;
                   2758:        }
                   2759: 
                   2760:       else if (ch == '\0')
                   2761:        error ("Null character found in input");
                   2762: 
                   2763:       else if (!comment_p)
                   2764:        {
                   2765:          if (ch == '"')
                   2766:            string_p = !string_p;
                   2767: 
                   2768:          else if (ch == '#')
                   2769:            comment_p++;
                   2770: 
                   2771:          else if (ch == ';' && !string_p)
                   2772:            {
                   2773:              line_split_p = 1;
                   2774:              *ptr++ = '\n';
                   2775:              *ptr = '\0';
                   2776:              cur_line_ptr = cur_line_start;
                   2777:              return cur_line_ptr;
                   2778:            }
                   2779:        }
                   2780:     }
                   2781: 
                   2782:   if (ferror (stdin))
                   2783:     pfatal_with_name (input_name);
                   2784: 
                   2785:   cur_line_ptr = (char *)0;
                   2786:   return (char *)0;
                   2787: }
                   2788: 
                   2789: 
                   2790: /* Parse #.begin directives which have a label as the first argument
                   2791:    which gives the location of the start of the block.  */
                   2792: 
                   2793: STATIC void
                   2794: parse_begin (start)
                   2795:      const char *start;                        /* start of directive */
                   2796: {
                   2797:   const char *end_p1;                  /* end of label */
                   2798:   int ch;
                   2799:   shash_t *hash_ptr;                   /* hash pointer to lookup label */
                   2800: 
                   2801:   if (cur_file_ptr == (efdr_t *)0)
                   2802:     {
                   2803:       error ("#.begin directive without a preceding .file directive");
                   2804:       return;
                   2805:     }
                   2806: 
                   2807:   if (cur_proc_ptr == (PDR *)0)
                   2808:     {
                   2809:       error ("#.begin directive without a preceding .ent directive");
                   2810:       return;
                   2811:     }
                   2812: 
                   2813:   for (end_p1 = start; (ch = *end_p1) != '\0' && !isspace (ch); end_p1++)
                   2814:     ;
                   2815: 
                   2816:   hash_ptr = hash_string (start,
                   2817:                          end_p1 - start,
                   2818:                          &orig_str_hash[0],
                   2819:                          (symint_t *)0);
                   2820: 
                   2821:   if (hash_ptr == (shash_t *)0)
                   2822:     {
                   2823:       error ("Label %.*s not found for #.begin", end_p1 - start, start);
                   2824:       return;
                   2825:     }
                   2826: 
                   2827:   if (cur_oproc_begin == (SYMR *)0)
                   2828:     {
                   2829:       error ("Procedure table %.*s not found for #.begin", end_p1 - start, start);
                   2830:       return;
                   2831:     }
                   2832: 
                   2833:   (void) add_local_symbol ((const char *)0, (const char *)0,
                   2834:                           st_Block, sc_Text,
                   2835:                           (symint_t)hash_ptr->sym_ptr->value - cur_oproc_begin->value,
                   2836:                           (symint_t)0);
                   2837: }
                   2838: 
                   2839: 
                   2840: /* Parse #.bend directives which have a label as the first argument
                   2841:    which gives the location of the end of the block.  */
                   2842: 
                   2843: STATIC void
                   2844: parse_bend (start)
                   2845:      const char *start;                        /* start of directive */
                   2846: {
                   2847:   const char *end_p1;                  /* end of label */
                   2848:   int ch;
                   2849:   shash_t *hash_ptr;                   /* hash pointer to lookup label */
                   2850: 
                   2851:   if (cur_file_ptr == (efdr_t *)0)
                   2852:     {
                   2853:       error ("#.begin directive without a preceding .file directive");
                   2854:       return;
                   2855:     }
                   2856: 
                   2857:   if (cur_proc_ptr == (PDR *)0)
                   2858:     {
                   2859:       error ("#.bend directive without a preceding .ent directive");
                   2860:       return;
                   2861:     }
                   2862: 
                   2863:   for (end_p1 = start; (ch = *end_p1) != '\0' && !isspace (ch); end_p1++)
                   2864:     ;
                   2865: 
                   2866:   hash_ptr = hash_string (start,
                   2867:                          end_p1 - start,
                   2868:                          &orig_str_hash[0],
                   2869:                          (symint_t *)0);
                   2870: 
                   2871:   if (hash_ptr == (shash_t *)0)
                   2872:     {
                   2873:       error ("Label %.*s not found for #.bend", end_p1 - start, start);
                   2874:       return;
                   2875:     }
                   2876: 
                   2877:   if (cur_oproc_begin == (SYMR *)0)
                   2878:     {
                   2879:       error ("Procedure table %.*s not found for #.bend", end_p1 - start, start);
                   2880:       return;
                   2881:     }
                   2882: 
                   2883:   (void) add_local_symbol ((const char *)0, (const char *)0,
                   2884:                           st_End, sc_Text,
                   2885:                           (symint_t)hash_ptr->sym_ptr->value - cur_oproc_begin->value,
                   2886:                           (symint_t)0);
                   2887: }
                   2888: 
                   2889: 
                   2890: /* Parse #.def directives, which are contain standard COFF subdirectives
                   2891:    to describe the debugging format.  These subdirectives include:
                   2892: 
                   2893:        .scl    specify storage class
                   2894:        .val    specify a value
                   2895:        .endef  specify end of COFF directives
                   2896:        .type   specify the type
                   2897:        .size   specify the size of an array
                   2898:        .dim    specify an array dimension
                   2899:        .tag    specify a tag for a struct, union, or enum.  */
                   2900: 
                   2901: STATIC void
                   2902: parse_def (name_start)
                   2903:      const char *name_start;                   /* start of directive */
                   2904: {
                   2905:   const char *dir_start;                       /* start of current directive*/
                   2906:   const char *dir_end_p1;                      /* end+1 of current directive*/
                   2907:   const char *arg_start;                       /* start of current argument */
                   2908:   const char *arg_end_p1;                      /* end+1 of current argument */
                   2909:   const char *name_end_p1;                     /* end+1 of label */
                   2910:   const char *tag_start          = (const char *)0;    /* start of tag name */
                   2911:   const char *tag_end_p1  = (const char *)0;   /* end+1 of tag name */
                   2912:   sc_t storage_class     = sc_Nil;
                   2913:   st_t symbol_type       = st_Nil;
                   2914:   type_info_t t;
                   2915:   EXTR *eptr             = (EXTR *)0;          /* ext. sym equivalent to def*/
                   2916:   int is_function        = 0;                  /* != 0 if function */
                   2917:   symint_t value         = 0;
                   2918:   symint_t indx                  = cur_file_ptr->void_type;
                   2919:   int error_line         = 0;
                   2920:   symint_t arg_number;
                   2921:   symint_t temp_array[ N_TQ ];
                   2922:   int arg_was_number;
                   2923:   int ch, i;
                   2924:   Ptrdiff_t len;
                   2925: 
                   2926:   static int inside_enumeration = 0;           /* is this an enumeration? */
                   2927: 
                   2928: 
                   2929:   /* Initialize the type information.  */
                   2930:   t = type_info_init;
                   2931: 
                   2932: 
                   2933:   /* Search for the end of the name being defined.  */
                   2934:   /* Allow spaces and such in names for G++ templates, which produce stabs
                   2935:      that look like:
                   2936: 
                   2937:      #.def   SMANIP<long unsigned int>; .scl 10; .type 0x8; .size 8; .endef */
                   2938: 
                   2939:   for (name_end_p1 = name_start; (ch = *name_end_p1) != ';' && ch != '\0'; name_end_p1++)
                   2940:     ;
                   2941: 
                   2942:   if (ch == '\0')
                   2943:     {
                   2944:       error_line = __LINE__;
                   2945:       saber_stop ();
                   2946:       goto bomb_out;
                   2947:     }
                   2948: 
                   2949:   /* Parse the remaining subdirectives now.  */
                   2950:   dir_start = name_end_p1+1;
                   2951:   for (;;)
                   2952:     {
                   2953:       while ((ch = *dir_start) == ' ' || ch == '\t')
                   2954:        ++dir_start;
                   2955: 
                   2956:       if (ch != '.')
                   2957:        {
                   2958:          error_line = __LINE__;
                   2959:          saber_stop ();
                   2960:          goto bomb_out;
                   2961:        }
                   2962: 
                   2963:       /* Are we done? */
                   2964:       if (dir_start[1] == 'e'
                   2965:          && memcmp (dir_start, ".endef", sizeof (".endef")-1) == 0)
                   2966:        break;
                   2967: 
                   2968:       /* Pick up the subdirective now */
                   2969:       for (dir_end_p1 = dir_start+1;
                   2970:           (ch = *dir_end_p1) != ' ' && ch != '\t';
                   2971:           dir_end_p1++)
                   2972:        {
                   2973:          if (ch == '\0' || isspace (ch))
                   2974:            {
                   2975:              error_line = __LINE__;
                   2976:              saber_stop ();
                   2977:              goto bomb_out;
                   2978:            }
                   2979:        }
                   2980: 
                   2981:       /* Pick up the subdirective argument now.  */
                   2982:       arg_was_number = arg_number = 0;
                   2983:       arg_end_p1 = (const char *)0;
                   2984:       arg_start = dir_end_p1+1;
                   2985:       ch = *arg_start;
                   2986:       while (ch == ' ' || ch == '\t')
                   2987:        ch = *++arg_start;
                   2988: 
                   2989:       if (isdigit (ch) || ch == '-' || ch == '+')
                   2990:        {
                   2991:          int ch2;
                   2992:          arg_number = strtol (arg_start, (char **) &arg_end_p1, 0);
                   2993:          if (arg_end_p1 != arg_start || (ch2 = *arg_end_p1 != ';') || ch2 != ',')
                   2994:            arg_was_number++;
                   2995:        }
                   2996: 
                   2997:       else if (ch == '\0' || isspace (ch))
                   2998:        {
                   2999:          error_line = __LINE__;
                   3000:          saber_stop ();
                   3001:          goto bomb_out;
                   3002:        }
                   3003: 
                   3004:       if (!arg_was_number)
                   3005:        {
                   3006:          /* Allow spaces and such in names for G++ templates.  */
                   3007:          for (arg_end_p1 = arg_start+1;
                   3008:               (ch = *arg_end_p1) != ';' && ch != '\0';
                   3009:               arg_end_p1++)
                   3010:            ;
                   3011: 
                   3012:          if (ch == '\0')
                   3013:            {
                   3014:              error_line = __LINE__;
                   3015:              saber_stop ();
                   3016:              goto bomb_out;
                   3017:            }
                   3018:        }
                   3019: 
                   3020:       /* Classify the directives now.  */
                   3021:       len = dir_end_p1 - dir_start;
                   3022:       switch (dir_start[1])
                   3023:        {
                   3024:        default:
                   3025:          error_line = __LINE__;
                   3026:          saber_stop ();
                   3027:          goto bomb_out;
                   3028: 
                   3029:        case 'd':
                   3030:          if (len == sizeof (".dim")-1
                   3031:              && memcmp (dir_start, ".dim", sizeof (".dim")-1) == 0
                   3032:              && arg_was_number)
                   3033:            {
                   3034:              symint_t *t_ptr = &temp_array[ N_TQ-1 ];
                   3035: 
                   3036:              *t_ptr = arg_number;
                   3037:              while (*arg_end_p1 == ',' && arg_was_number)
                   3038:                {
                   3039:                  arg_start = arg_end_p1+1;
                   3040:                  ch = *arg_start;
                   3041:                  while (ch == ' ' || ch == '\t')
                   3042:                    ch = *++arg_start;
                   3043: 
                   3044:                  arg_was_number = 0;
                   3045:                  if (isdigit (ch) || ch == '-' || ch == '+')
                   3046:                    {
                   3047:                      int ch2;
                   3048:                      arg_number = strtol (arg_start, (char **) &arg_end_p1, 0);
                   3049:                      if (arg_end_p1 != arg_start || (ch2 = *arg_end_p1 != ';') || ch2 != ',')
                   3050:                        arg_was_number++;
                   3051: 
                   3052:                      if (t_ptr == &temp_array[0])
                   3053:                        {
                   3054:                          error_line = __LINE__;
                   3055:                          saber_stop ();
                   3056:                          goto bomb_out;
                   3057:                        }
                   3058: 
                   3059:                      *--t_ptr = arg_number;
                   3060:                    }
                   3061:                }
                   3062: 
                   3063:              /* Reverse order of dimensions.  */
                   3064:              while (t_ptr <= &temp_array[ N_TQ-1 ])
                   3065:                {
                   3066:                  if (t.num_dims >= N_TQ-1)
                   3067:                    {
                   3068:                      error_line = __LINE__;
                   3069:                      saber_stop ();
                   3070:                      goto bomb_out;
                   3071:                    }
                   3072: 
                   3073:                  t.dimensions[ t.num_dims++ ] = *t_ptr++;
                   3074:                }
                   3075:              break;
                   3076:            }
                   3077:          else
                   3078:            {
                   3079:              error_line = __LINE__;
                   3080:              saber_stop ();
                   3081:              goto bomb_out;
                   3082:            }
                   3083: 
                   3084: 
                   3085:        case 's':
                   3086:          if (len == sizeof (".scl")-1
                   3087:              && memcmp (dir_start, ".scl", sizeof (".scl")-1) == 0
                   3088:              && arg_was_number
                   3089:              && arg_number < ((symint_t) C_MAX))
                   3090:            {
                   3091:              /* If the symbol is a static or external, we have
                   3092:                 already gotten the appropriate type and class, so
                   3093:                 make sure we don't override those values.  This is
                   3094:                 needed because there are some type and classes that
                   3095:                 are not in COFF, such as short data, etc.  */
                   3096:              if (symbol_type == st_Nil)
                   3097:                {
                   3098:                  symbol_type   = map_coff_sym_type[arg_number];
                   3099:                  storage_class = map_coff_storage [arg_number];
                   3100:                }
                   3101:              break;
                   3102:            }
                   3103: 
                   3104:          else if (len == sizeof (".size")-1
                   3105:                   && memcmp (dir_start, ".size", sizeof (".size")-1) == 0
                   3106:                   && arg_was_number)
                   3107:            {
                   3108:              symint_t *t_ptr = &temp_array[ N_TQ-1 ];
                   3109: 
                   3110:              *t_ptr = arg_number;
                   3111:              while (*arg_end_p1 == ',' && arg_was_number)
                   3112:                {
                   3113:                  arg_start = arg_end_p1+1;
                   3114:                  ch = *arg_start;
                   3115:                  while (ch == ' ' || ch == '\t')
                   3116:                    ch = *++arg_start;
                   3117: 
                   3118:                  arg_was_number = 0;
                   3119:                  if (isdigit (ch) || ch == '-' || ch == '+')
                   3120:                    {
                   3121:                      int ch2;
                   3122:                      arg_number = strtol (arg_start, (char **) &arg_end_p1, 0);
                   3123:                      if (arg_end_p1 != arg_start || (ch2 = *arg_end_p1 != ';') || ch2 != ',')
                   3124:                        arg_was_number++;
                   3125: 
                   3126:                      if (t_ptr == &temp_array[0])
                   3127:                        {
                   3128:                          error_line = __LINE__;
                   3129:                          saber_stop ();
                   3130:                          goto bomb_out;
                   3131:                        }
                   3132: 
                   3133:                      *--t_ptr = arg_number;
                   3134:                    }
                   3135:                }
                   3136: 
                   3137:              /* Reverse order of sizes.  */
                   3138:              while (t_ptr <= &temp_array[ N_TQ-1 ])
                   3139:                {
                   3140:                  if (t.num_sizes >= N_TQ-1)
                   3141:                    {
                   3142:                      error_line = __LINE__;
                   3143:                      saber_stop ();
                   3144:                      goto bomb_out;
                   3145:                    }
                   3146: 
                   3147:                  t.sizes[ t.num_sizes++ ] = *t_ptr++;
                   3148:                }
                   3149:              break;
                   3150:            }
                   3151: 
                   3152:          else
                   3153:            {
                   3154:              error_line = __LINE__;
                   3155:              saber_stop ();
                   3156:              goto bomb_out;
                   3157:            }
                   3158: 
                   3159: 
                   3160:        case 't':
                   3161:          if (len == sizeof (".type")-1
                   3162:              && memcmp (dir_start, ".type", sizeof (".type")-1) == 0
                   3163:              && arg_was_number)
                   3164:            {
                   3165:              tq_t *tq_ptr = &t.type_qualifiers[0];
                   3166: 
                   3167:              t.orig_type = (coff_type_t) (arg_number & N_BTMASK);
                   3168:              t.basic_type = map_coff_types [(int)t.orig_type];
                   3169:              for (i = N_TQ-1; i >= 0; i--)
                   3170:                {
                   3171:                  int dt = (arg_number >> ((i * N_TQ_SHIFT) + N_BT_SHIFT)
                   3172:                            & N_TMASK);
                   3173: 
                   3174:                  if (dt != (int)DT_NON)
                   3175:                    *tq_ptr++ = map_coff_derived_type [dt];
                   3176:                }
                   3177: 
                   3178:              /* If this is a function, ignore it, so that we don't get
                   3179:                 two entries (one from the .ent, and one for the .def
                   3180:                 that precedes it).  Save the type information so that
                   3181:                 the end block can properly add it after the begin block
                   3182:                 index.  For MIPS knows what reason, we must strip off
                   3183:                 the function type at this point.  */
                   3184:              if (tq_ptr != &t.type_qualifiers[0] && tq_ptr[-1] == tq_Proc)
                   3185:                {
                   3186:                  is_function = 1;
                   3187:                  tq_ptr[-1] = tq_Nil;
                   3188:                }
                   3189: 
                   3190:              break;
                   3191:            }
                   3192: 
                   3193:          else if (len == sizeof (".tag")-1
                   3194:              && memcmp (dir_start, ".tag", sizeof (".tag")-1) == 0)
                   3195:            {
                   3196:              tag_start = arg_start;
                   3197:              tag_end_p1 = arg_end_p1;
                   3198:              break;
                   3199:            }
                   3200: 
                   3201:          else
                   3202:            {
                   3203:              error_line = __LINE__;
                   3204:              saber_stop ();
                   3205:              goto bomb_out;
                   3206:            }
                   3207: 
                   3208: 
                   3209:        case 'v':
                   3210:          if (len == sizeof (".val")-1
                   3211:              && memcmp (dir_start, ".val", sizeof (".val")-1) == 0)
                   3212:            {
                   3213:              if (arg_was_number)
                   3214:                value = arg_number;
                   3215: 
                   3216:              /* If the value is not an integer value, it must be the
                   3217:                 name of a static or global item.  Look up the name in
                   3218:                 the original symbol table to pick up the storage
                   3219:                 class, symbol type, etc.  */
                   3220:              else
                   3221:                {
                   3222:                  shash_t *orig_hash_ptr;       /* hash within orig sym table*/
                   3223:                  shash_t *ext_hash_ptr;        /* hash within ext. sym table*/
                   3224: 
                   3225:                  ext_hash_ptr = hash_string (arg_start,
                   3226:                                              arg_end_p1 - arg_start,
                   3227:                                              &ext_str_hash[0],
                   3228:                                              (symint_t *)0);
                   3229: 
                   3230:                  if (ext_hash_ptr != (shash_t *)0
                   3231:                      && ext_hash_ptr->esym_ptr != (EXTR *)0)
                   3232:                    eptr = ext_hash_ptr->esym_ptr;
                   3233: 
                   3234:                  orig_hash_ptr = hash_string (arg_start,
                   3235:                                               arg_end_p1 - arg_start,
                   3236:                                               &orig_str_hash[0],
                   3237:                                               (symint_t *)0);
                   3238: 
                   3239:                  if ((orig_hash_ptr == (shash_t *)0
                   3240:                       || orig_hash_ptr->sym_ptr == (SYMR *)0)
                   3241:                      && eptr == (EXTR *)0)
                   3242:                    {
                   3243:                      fprintf (stderr, "warning, %.*s not found in original or external symbol tables, value defaults to 0\n",
                   3244:                               arg_end_p1 - arg_start,
                   3245:                               arg_start);
                   3246:                      value = 0;
                   3247:                    }
                   3248:                  else
                   3249:                    {
                   3250:                      SYMR *ptr = (orig_hash_ptr != (shash_t *)0
                   3251:                                   && orig_hash_ptr->sym_ptr != (SYMR *)0)
                   3252:                                        ? orig_hash_ptr->sym_ptr
                   3253:                                        : &eptr->asym;
                   3254: 
                   3255:                      symbol_type = (st_t) ptr->st;
                   3256:                      storage_class = (sc_t) ptr->sc;
                   3257:                      value = ptr->value;
                   3258:                    }
                   3259:                }
                   3260:              break;
                   3261:            }
                   3262:          else
                   3263:            {
                   3264:              error_line = __LINE__;
                   3265:              saber_stop ();
                   3266:              goto bomb_out;
                   3267:            }
                   3268:        }
                   3269: 
                   3270:       /* Set up to find next directive.  */
                   3271:       dir_start = arg_end_p1 + 1;
                   3272:     }
                   3273: 
                   3274: 
                   3275:   t.extra_sizes = (tag_start != (char *)0);
                   3276:   if (t.num_dims > 0)
                   3277:     {
                   3278:       int diff = t.num_dims - t.num_sizes;
                   3279:       int i = t.num_dims - 1;
                   3280:       int j;
                   3281: 
                   3282:       if (t.num_sizes != 1 || diff < 0)
                   3283:        {
                   3284:          error_line = __LINE__;
                   3285:          saber_stop ();
                   3286:          goto bomb_out;
                   3287:        }
                   3288: 
                   3289:       /* If this is an array, make sure the same number of dimensions
                   3290:         and sizes were passed, creating extra sizes for multiply
                   3291:         dimensioned arrays if not passed.  */
                   3292: 
                   3293:       t.extra_sizes = 0;
                   3294:       if (diff)
                   3295:        {
                   3296:          for (j = (sizeof (t.sizes) / sizeof (t.sizes[0])) - 1; j >= 0; j--)
                   3297:            t.sizes[ j ] = ((j-diff) >= 0) ? t.sizes[ j-diff ] : 0;
                   3298: 
                   3299:          t.num_sizes = i + 1;
                   3300:          for ( i--; i >= 0; i-- )
                   3301:            {
                   3302:              if (t.dimensions[ i+1 ])
                   3303:                t.sizes[ i ] = t.sizes[ i+1 ] / t.dimensions[ i+1 ];
                   3304:              else
                   3305:                t.sizes[ i ] = t.sizes[ i+1 ];
                   3306:            }
                   3307:        }
                   3308:     }
                   3309: 
                   3310:   else if (symbol_type == st_Member && t.num_sizes - t.extra_sizes == 1)
                   3311:     { /* Is this a bitfield?  This is indicated by a structure memeber
                   3312:         having a size field that isn't an array.  */
                   3313: 
                   3314:       t.bitfield = 1;
                   3315:     }
                   3316: 
                   3317: 
                   3318:   /* Except for enumeration members & begin/ending of scopes, put the
                   3319:      type word in the aux. symbol table.  */
                   3320: 
                   3321:   if (symbol_type == st_Block || symbol_type == st_End)
                   3322:     indx = 0;
                   3323: 
                   3324:   else if (inside_enumeration)
                   3325:     indx = cur_file_ptr->void_type;
                   3326: 
                   3327:   else
                   3328:     {
                   3329:       if (t.basic_type == bt_Struct
                   3330:          || t.basic_type == bt_Union
                   3331:          || t.basic_type == bt_Enum)
                   3332:        {
                   3333:          if (tag_start == (char *)0)
                   3334:            {
                   3335:              error ("No tag specified for %.*s",
                   3336:                     name_end_p1 - name_start,
                   3337:                     name_start);
                   3338:              return;
                   3339:            }
                   3340: 
                   3341:          t.tag_ptr = get_tag (tag_start, tag_end_p1,  (symint_t)indexNil,
                   3342:                               t.basic_type);
                   3343:        }
                   3344: 
                   3345:       if (is_function)
                   3346:        {
                   3347:          last_func_type_info = t;
                   3348:          last_func_eptr = eptr;
                   3349:          return;
                   3350:        }
                   3351: 
                   3352:       indx = add_aux_sym_tir (&t,
                   3353:                              hash_yes,
                   3354:                              &cur_file_ptr->thash_head[0]);
                   3355:     }
                   3356: 
                   3357: 
                   3358:   /* If this is an external or static symbol, update the appropriate
                   3359:      external symbol.  */
                   3360: 
                   3361:   if (eptr != (EXTR *)0
                   3362:       && (eptr->asym.index == indexNil || cur_proc_ptr == (PDR *)0))
                   3363:     {
                   3364:       eptr->ifd = cur_file_ptr->file_index;
                   3365:       eptr->asym.index = indx;
                   3366:     }
                   3367: 
                   3368: 
                   3369:   /* Do any last minute adjustments that are necessary.  */
                   3370:   switch (symbol_type)
                   3371:     {
                   3372:     default:
                   3373:       break;
                   3374: 
                   3375: 
                   3376:       /* For the beginning of structs, unions, and enumerations, the
                   3377:         size info needs to be passed in the value field.  */
                   3378: 
                   3379:     case st_Block:
                   3380:       if (t.num_sizes - t.num_dims - t.extra_sizes != 1)
                   3381:        {
                   3382:          error_line = __LINE__;
                   3383:          saber_stop ();
                   3384:          goto bomb_out;
                   3385:        }
                   3386: 
                   3387:       else
                   3388:        value = t.sizes[0];
                   3389: 
                   3390:       inside_enumeration = (t.orig_type == T_ENUM);
                   3391:       break;
                   3392: 
                   3393: 
                   3394:       /* For the end of structs, unions, and enumerations, omit the
                   3395:         name which is always ".eos".  This needs to be done last, so
                   3396:         that any error reporting above gives the correct name.  */
                   3397: 
                   3398:     case st_End:
                   3399:       name_start = name_end_p1 = (const char *)0;
                   3400:       value = inside_enumeration = 0;
                   3401:       break;
                   3402: 
                   3403: 
                   3404:       /* Members of structures and unions that aren't bitfields, need
                   3405:         to adjust the value from a byte offset to a bit offset.
                   3406:         Members of enumerations do not have the value adjusted, and
                   3407:         can be distinguished by indx == indexNil.  For enumerations,
                   3408:         update the maximum enumeration value.  */
                   3409: 
                   3410:     case st_Member:
                   3411:       if (!t.bitfield && !inside_enumeration)
                   3412:        value *= 8;
                   3413: 
                   3414:       break;
                   3415:     }
                   3416: 
                   3417: 
                   3418:   /* Add the symbol, except for global symbols outside of functions,
                   3419:      for which the external symbol table is fine enough.  */
                   3420: 
                   3421:   if (eptr == (EXTR *)0
                   3422:       || eptr->asym.st == (int)st_Nil
                   3423:       || cur_proc_ptr != (PDR *)0)
                   3424:     {
                   3425:       symint_t isym = add_local_symbol (name_start, name_end_p1,
                   3426:                                        symbol_type, storage_class,
                   3427:                                        value,
                   3428:                                        indx);
                   3429: 
                   3430:       /* deal with struct, union, and enum tags.  */
                   3431:       if (symbol_type == st_Block)
                   3432:         {
                   3433:          /* Create or update the tag information.  */
                   3434:          tag_t *tag_ptr = get_tag (name_start,
                   3435:                                    name_end_p1,
                   3436:                                    isym,
                   3437:                                    t.basic_type);
                   3438: 
                   3439:          /* If there are any forward references, fill in the appropriate
                   3440:             file and symbol indexes.  */
                   3441: 
                   3442:          symint_t file_index  = cur_file_ptr->file_index;
                   3443:          forward_t *f_next = tag_ptr->forward_ref;
                   3444:          forward_t *f_cur;
                   3445: 
                   3446:          while (f_next != (forward_t *)0)
                   3447:            {
                   3448:              f_cur  = f_next;
                   3449:              f_next = f_next->next;
                   3450: 
                   3451:              f_cur->ifd_ptr->isym = file_index;
                   3452:              f_cur->index_ptr->rndx.index = isym;
                   3453: 
                   3454:              free_forward (f_cur);
                   3455:            }
                   3456: 
                   3457:          tag_ptr->forward_ref = (forward_t *)0;
                   3458:         }
                   3459:     }
                   3460: 
                   3461:   /* Normal return  */
                   3462:   return;
                   3463: 
                   3464:   /* Error return, issue message.  */
                   3465: bomb_out:
                   3466:   if (error_line)
                   3467:     error ("compiler error, badly formed #.def (internal line # = %d)", error_line);
                   3468:   else
                   3469:     error ("compiler error, badly formed #.def");
                   3470: 
                   3471:   return;
                   3472: }
                   3473: 
                   3474: 
                   3475: /* Parse .end directives.  */
                   3476: 
                   3477: STATIC void
                   3478: parse_end (start)
                   3479:      const char *start;                        /* start of directive */
                   3480: {
                   3481:   register const char *start_func, *end_func_p1;
                   3482:   register int ch;
                   3483:   register symint_t value;
                   3484:   register FDR *orig_fdr;
                   3485: 
                   3486:   if (cur_file_ptr == (efdr_t *)0)
                   3487:     {
                   3488:       error (".end directive without a preceding .file directive");
                   3489:       return;
                   3490:     }
                   3491: 
                   3492:   if (cur_proc_ptr == (PDR *)0)
                   3493:     {
                   3494:       error (".end directive without a preceding .ent directive");
                   3495:       return;
                   3496:     }
                   3497: 
                   3498:   /* Get the function name, skipping whitespace.  */
                   3499:   for (start_func = start; isspace (*start_func); start_func++)
                   3500:     ;
                   3501: 
                   3502:   ch = *start_func;
                   3503:   if (!IS_ASM_IDENT (ch))
                   3504:     {
                   3505:       error (".end directive has no name");
                   3506:       return;
                   3507:     }
                   3508: 
                   3509:   for (end_func_p1 = start_func; IS_ASM_IDENT (ch); ch = *++end_func_p1)
                   3510:     ;
                   3511: 
                   3512: 
                   3513:   /* Get the value field for creating the end from the original object
                   3514:      file (which we find by locating the procedure start, and using the
                   3515:      pointer to the end+1 block and backing up.  The index points to a
                   3516:      two word aux. symbol, whose first word is the index of the end
                   3517:      symbol, and the second word is the type of the function return
                   3518:      value.  */
                   3519: 
                   3520:   orig_fdr = cur_file_ptr->orig_fdr;
                   3521:   value = 0;
                   3522:   if (orig_fdr != (FDR *)0 && cur_oproc_end != (SYMR *)0)
                   3523:     value = cur_oproc_end->value;
                   3524: 
                   3525:   else
                   3526:     error ("Cannot find .end block for %.*s", end_func_p1 - start_func, start_func);
                   3527: 
                   3528:   (void) add_local_symbol (start_func, end_func_p1,
                   3529:                           st_End, sc_Text,
                   3530:                           value,
                   3531:                           (symint_t)0);
                   3532: 
                   3533:   cur_proc_ptr = cur_oproc_ptr = (PDR *)0;
                   3534: }
                   3535: 
                   3536: 
                   3537: /* Parse .ent directives.  */
                   3538: 
                   3539: STATIC void
                   3540: parse_ent (start)
                   3541:      const char *start;                        /* start of directive */
                   3542: {
                   3543:   register const char *start_func, *end_func_p1;
                   3544:   register int ch;
                   3545: 
                   3546:   if (cur_file_ptr == (efdr_t *)0)
                   3547:     {
                   3548:       error (".ent directive without a preceding .file directive");
                   3549:       return;
                   3550:     }
                   3551: 
                   3552:   if (cur_proc_ptr != (PDR *)0)
                   3553:     {
                   3554:       error ("second .ent directive found before .end directive");
                   3555:       return;
                   3556:     }
                   3557: 
                   3558:   for (start_func = start; isspace (*start_func); start_func++)
                   3559:     ;
                   3560: 
                   3561:   ch = *start_func;
                   3562:   if (!IS_ASM_IDENT (ch))
                   3563:     {
                   3564:       error (".ent directive has no name");
                   3565:       return;
                   3566:     }
                   3567: 
                   3568:   for (end_func_p1 = start_func; IS_ASM_IDENT (ch); ch = *++end_func_p1)
                   3569:     ;
                   3570: 
                   3571:   (void) add_procedure (start_func, end_func_p1);
                   3572: }
                   3573: 
                   3574: 
                   3575: /* Parse .file directives.  */
                   3576: 
                   3577: STATIC void
                   3578: parse_file (start)
                   3579:      const char *start;                        /* start of directive */
                   3580: {
                   3581:   char *p;
                   3582:   register char *start_name, *end_name_p1;
                   3583: 
                   3584:   (void) strtol (start, &p, 0);
                   3585:   if (start == p
                   3586:       || (start_name = local_index (p, '"')) == (char *)0
                   3587:       || (end_name_p1 = local_rindex (++start_name, '"')) == (char *)0)
                   3588:     {
                   3589:       error ("Invalid .file directive");
                   3590:       return;
                   3591:     }
                   3592: 
                   3593:   if (cur_proc_ptr != (PDR *)0)
                   3594:     {
                   3595:       error ("No way to handle .file within .ent/.end section");
                   3596:       return;
                   3597:     }
                   3598: 
                   3599:   add_file (start_name, end_name_p1);
                   3600: }
                   3601: 
                   3602: 
                   3603: /* Make sure the @stabs symbol is emitted.  */
                   3604: 
                   3605: static void
                   3606: mark_stabs (start)
                   3607:      const char *start;                        /* Start of directive (ignored) */
                   3608: {
                   3609:   if (!stabs_seen)
                   3610:     {
                   3611:       /* Add a dummy @stabs dymbol. */
                   3612:       stabs_seen = 1;
                   3613:       (void) add_local_symbol (stabs_symbol,
                   3614:                               stabs_symbol + sizeof (stabs_symbol),
                   3615:                               stNil, scInfo, -1, MIPS_MARK_STAB(0));
                   3616: 
                   3617:     }
                   3618: }
                   3619: 
                   3620: 
                   3621: /* Parse .stabs directives.
                   3622: 
                   3623:    .stabs directives have five fields:
                   3624:        "string"        a string, encoding the type information.
                   3625:        code            a numeric code, defined in <stab.h>
                   3626:        0               a zero
                   3627:        0               a zero or line number
                   3628:        value           a numeric value or an address.
                   3629: 
                   3630:     If the value is relocatable, we transform this into:
                   3631:        iss             points as an index into string space
                   3632:        value           value from lookup of the name
                   3633:        st              st from lookup of the name
                   3634:        sc              sc from lookup of the name
                   3635:        index           code|CODE_MASK
                   3636: 
                   3637:     If the value is not relocatable, we transform this into:
                   3638:        iss             points as an index into string space
                   3639:        value           value
                   3640:        st              st_Nil
                   3641:        sc              sc_Nil
                   3642:        index           code|CODE_MASK
                   3643: 
                   3644:     .stabn directives have four fields (string is null):
                   3645:        code            a numeric code, defined in <stab.h>
                   3646:        0               a zero
                   3647:        0               a zero or a line number
                   3648:        value           a numeric value or an address.  */
                   3649: 
                   3650: STATIC void
                   3651: parse_stabs_common (string_start, string_end, rest)
                   3652:      const char *string_start;         /* start of string or NULL */
                   3653:      const char *string_end;           /* end+1 of string or NULL */
                   3654:      const char *rest;                 /* rest of the directive. */
                   3655: {
                   3656:   efdr_t *save_file_ptr = cur_file_ptr;
                   3657:   symint_t code;
                   3658:   symint_t value;
                   3659:   char *p;
                   3660:   st_t st;
                   3661:   sc_t sc;
                   3662:   int ch;
                   3663: 
                   3664:   if (stabs_seen == 0)
                   3665:     mark_stabs ("");
                   3666: 
                   3667:   /* Read code from stabs.  */
                   3668:   if (!isdigit (*rest))
                   3669:     {
                   3670:       error ("Invalid .stabs/.stabn directive, code is non-numeric");
                   3671:       return;
                   3672:     }
                   3673: 
                   3674:   code = strtol (rest, &p, 0);
                   3675: 
                   3676:   /* Line number stabs are handled differently, since they have two values,
                   3677:      the line number and the address of the label.  We use the index field
                   3678:      (aka code) to hold the line number, and the value field to hold the
                   3679:      address.  The symbol type is st_Label, which should be different from
                   3680:      the other stabs, so that gdb can recognize it.  */
                   3681: 
                   3682:   if (code == (int)N_SLINE)
                   3683:     {
                   3684:       SYMR *sym_ptr, dummy_symr;
                   3685:       shash_t *shash_ptr;
                   3686: 
                   3687:       /* Skip ,0, */
                   3688:       if (p[0] != ',' || p[1] != '0' || p[2] != ',' || !isdigit (p[3]))
                   3689:        {
                   3690:          error ("Invalid line number .stabs/.stabn directive");
                   3691:          return;
                   3692:        }
                   3693: 
                   3694:       code = strtol (p+3, &p, 0);
                   3695:       ch = *++p;
                   3696:       if (p[-1] != ',' || isdigit (ch) || !IS_ASM_IDENT (ch))
                   3697:        {
                   3698:          error ("Invalid line number .stabs/.stabn directive");
                   3699:          return;
                   3700:        }
                   3701: 
                   3702:       dummy_symr.index = code;
                   3703:       if (dummy_symr.index != code)
                   3704:        {
                   3705:          error ("Line number (%d) for .stabs/.stabn directive cannot fit in index field (20 bits)",
                   3706:                 code);
                   3707: 
                   3708:          return;
                   3709:        }
                   3710: 
                   3711:       shash_ptr = hash_string (p,
                   3712:                               strlen (p) - 1,
                   3713:                               &orig_str_hash[0],
                   3714:                               (symint_t *)0);
                   3715: 
                   3716:       if (shash_ptr == (shash_t *)0
                   3717:          || (sym_ptr = shash_ptr->sym_ptr) == (SYMR *)0)
                   3718:        {
                   3719:          error ("Invalid .stabs/.stabn directive, value not found");
                   3720:          return;
                   3721:        }
                   3722: 
                   3723:       if ((st_t) sym_ptr->st != st_Label)
                   3724:        {
                   3725:          error ("Invalid line number .stabs/.stabn directive");
                   3726:          return;
                   3727:        }
                   3728: 
                   3729:       st = st_Label;
                   3730:       sc = (sc_t) sym_ptr->sc;
                   3731:       value = sym_ptr->value;
                   3732:     }
                   3733:   else
                   3734:     {
                   3735:       /* Skip ,<num>,<num>, */
                   3736:       if (*p++ != ',')
                   3737:        goto failure;
                   3738:       for (; isdigit (*p); p++)
                   3739:        ;
                   3740:       if (*p++ != ',')
                   3741:        goto failure;
                   3742:       for (; isdigit (*p); p++)
                   3743:        ;
                   3744:       if (*p++ != ',')
                   3745:        goto failure;
                   3746:       ch = *p;
                   3747:       if (!IS_ASM_IDENT (ch) && ch != '-')
                   3748:        {
                   3749:        failure:
                   3750:          error ("Invalid .stabs/.stabn directive, bad character");
                   3751:          return;
                   3752:        }
                   3753: 
                   3754:       if (isdigit (ch) || ch == '-')
                   3755:        {
                   3756:          st = st_Nil;
                   3757:          sc = sc_Nil;
                   3758:          value = strtol (p, &p, 0);
                   3759:          if (*p != '\n')
                   3760:            {
                   3761:              error ("Invalid .stabs/.stabn directive, stuff after numeric value");
                   3762:              return;
                   3763:            }
                   3764:        }
                   3765:       else if (!IS_ASM_IDENT (ch))
                   3766:        {
                   3767:          error ("Invalid .stabs/.stabn directive, bad character");
                   3768:          return;
                   3769:        }
                   3770:       else
                   3771:        {
                   3772:          SYMR *sym_ptr;
                   3773:          shash_t *shash_ptr;
                   3774:          const char *start, *end_p1;
                   3775: 
                   3776:          start = p;
                   3777:          if ((end_p1 = strchr (start, '+')) == (char *)0)
                   3778:            {
                   3779:              if ((end_p1 = strchr (start, '-')) == (char *)0)
                   3780:                end_p1 = start + strlen(start) - 1;
                   3781:            }
                   3782: 
                   3783:          shash_ptr = hash_string (start,
                   3784:                                   end_p1 - start,
                   3785:                                   &orig_str_hash[0],
                   3786:                                   (symint_t *)0);
                   3787: 
                   3788:          if (shash_ptr == (shash_t *)0
                   3789:              || (sym_ptr = shash_ptr->sym_ptr) == (SYMR *)0)
                   3790:            {
                   3791:              shash_ptr = hash_string (start,
                   3792:                                       end_p1 - start,
                   3793:                                       &ext_str_hash[0],
                   3794:                                       (symint_t *)0);
                   3795: 
                   3796:              if (shash_ptr == (shash_t *)0
                   3797:                  || shash_ptr->esym_ptr == (EXTR *)0)
                   3798:                {
                   3799:                  error ("Invalid .stabs/.stabn directive, value not found");
                   3800:                  return;
                   3801:                }
                   3802:              else
                   3803:                sym_ptr = &(shash_ptr->esym_ptr->asym);
                   3804:            }
                   3805: 
                   3806:          /* Traditionally, N_LBRAC and N_RBRAC are *not* relocated. */
                   3807:          if (code == (int)N_LBRAC || code == (int)N_RBRAC)
                   3808:            {
                   3809:              sc = scNil;
                   3810:              st = stNil;
                   3811:            }
                   3812:          else
                   3813:            {
                   3814:              sc = (sc_t) sym_ptr->sc;
                   3815:              st = (st_t) sym_ptr->st;
                   3816:            }
                   3817:          value = sym_ptr->value;
                   3818: 
                   3819:          ch = *end_p1++;
                   3820:          if (ch != '\n')
                   3821:            {
                   3822:              if (((!isdigit (*end_p1)) && (*end_p1 != '-'))
                   3823:                  || ((ch != '+') && (ch != '-')))
                   3824:                {
                   3825:                  error ("Invalid .stabs/.stabn directive, badly formed value");
                   3826:                  return;
                   3827:                }
                   3828:              if (ch == '+')
                   3829:                value += strtol (end_p1, &p, 0);
                   3830:              else if (ch == '-')
                   3831:                value -= strtol (end_p1, &p, 0);
                   3832: 
                   3833:              if (*p != '\n')
                   3834:                {
                   3835:                  error ("Invalid .stabs/.stabn directive, stuff after numeric value");
                   3836:                  return;
                   3837:                }
                   3838:            }
                   3839:        }
                   3840:       code = MIPS_MARK_STAB(code);
                   3841:     }
                   3842: 
                   3843:   (void) add_local_symbol (string_start, string_end, st, sc, value, code);
                   3844:   /* Restore normal file type.  */
                   3845:   cur_file_ptr = save_file_ptr;
                   3846: }
                   3847: 
                   3848: 
                   3849: STATIC void
                   3850: parse_stabs (start)
                   3851:      const char *start;                        /* start of directive */
                   3852: {
                   3853:   const char *end = local_index (start+1, '"');
                   3854: 
                   3855:   if (*start != '"' || end == (const char *)0 || end[1] != ',')
                   3856:     {
                   3857:       error ("Invalid .stabs directive, no string");
                   3858:       return;
                   3859:     }
                   3860: 
                   3861:   parse_stabs_common (start+1, end, end+2);
                   3862: }
                   3863: 
                   3864: 
                   3865: STATIC void
                   3866: parse_stabn (start)
                   3867:      const char *start;                        /* start of directive */
                   3868: {
                   3869:   parse_stabs_common ((const char *)0, (const char *)0, start);
                   3870: }
                   3871: 
                   3872: 
                   3873: /* Parse the input file, and write the lines to the output file
                   3874:    if needed.  */
                   3875: 
                   3876: STATIC void
                   3877: parse_input __proto((void))
                   3878: {
                   3879:   register char *p;
                   3880:   register int i;
                   3881:   register thead_t *ptag_head;
                   3882:   register tag_t *ptag;
                   3883:   register tag_t *ptag_next;
                   3884: 
                   3885:   if (debug)
                   3886:     fprintf (stderr, "\tinput\n");
                   3887: 
                   3888:   /* Add a dummy scope block around the entire compilation unit for
                   3889:      structures defined outside of blocks.  */
                   3890:   ptag_head = allocate_thead ();
                   3891:   ptag_head->first_tag = 0;
                   3892:   ptag_head->prev = cur_tag_head;
                   3893:   cur_tag_head = ptag_head;
                   3894: 
                   3895:   while ((p = read_line ()) != (char *)0)
                   3896:     {
                   3897:       /* Skip leading blanks */
                   3898:       while (isspace (*p))
                   3899:        p++;
                   3900: 
                   3901:       /* See if it's a directive we handle.  If so, dispatch handler.  */
                   3902:       for (i = 0; i < sizeof (pseudo_ops) / sizeof (pseudo_ops[0]); i++)
                   3903:        if (memcmp (p, pseudo_ops[i].name, pseudo_ops[i].len) == 0
                   3904:            && isspace (p[pseudo_ops[i].len]))
                   3905:          {
                   3906:            p += pseudo_ops[i].len;     /* skip to first argument */
                   3907:            while (isspace (*p))
                   3908:              p++;
                   3909: 
                   3910:            (*pseudo_ops[i].func)( p );
                   3911:            break;
                   3912:          }
                   3913:     }
                   3914: 
                   3915:   /* Process any tags at global level.  */
                   3916:   ptag_head = cur_tag_head;
                   3917:   cur_tag_head = ptag_head->prev;
                   3918: 
                   3919:   for (ptag = ptag_head->first_tag;
                   3920:        ptag != (tag_t *)0;
                   3921:        ptag = ptag_next)
                   3922:     {
                   3923:       if (ptag->forward_ref != (forward_t *)0)
                   3924:        add_unknown_tag (ptag);
                   3925: 
                   3926:       ptag_next = ptag->same_block;
                   3927:       ptag->hash_ptr->tag_ptr = ptag->same_name;
                   3928:       free_tag (ptag);
                   3929:     }
                   3930: 
                   3931:   free_thead (ptag_head);
                   3932: 
                   3933: }
                   3934: 
                   3935: 
                   3936: /* Update the global headers with the final offsets in preparation
                   3937:    to write out the .T file.  */
                   3938: 
                   3939: STATIC void
                   3940: update_headers __proto((void))
                   3941: {
                   3942:   register symint_t i;
                   3943:   register efdr_t *file_ptr;
                   3944: 
                   3945:   /* Set up the symbolic header.  */
                   3946:   file_offset = sizeof (symbolic_header) + orig_file_header.f_symptr;
                   3947:   symbolic_header.magic = orig_sym_hdr.magic;
                   3948:   symbolic_header.vstamp = orig_sym_hdr.vstamp;
                   3949: 
                   3950:   /* Set up global counts.  */
                   3951:   symbolic_header.issExtMax = ext_strings.num_allocated;
                   3952:   symbolic_header.idnMax    = dense_num.num_allocated;
                   3953:   symbolic_header.ifdMax    = file_desc.num_allocated;
                   3954:   symbolic_header.iextMax   = ext_symbols.num_allocated;
                   3955:   symbolic_header.ilineMax  = orig_sym_hdr.ilineMax;
                   3956:   symbolic_header.ioptMax   = orig_sym_hdr.ioptMax;
                   3957:   symbolic_header.cbLine    = orig_sym_hdr.cbLine;
                   3958:   symbolic_header.crfd      = orig_sym_hdr.crfd;
                   3959: 
                   3960: 
                   3961:   /* Loop through each file, figuring out how many local syms,
                   3962:      line numbers, etc. there are.  Also, put out end symbol
                   3963:      for the filename.  */
                   3964: 
                   3965:   for (file_ptr = first_file;
                   3966:        file_ptr != (efdr_t *)0;
                   3967:        file_ptr = file_ptr->next_file)
                   3968:     {
                   3969:       cur_file_ptr = file_ptr;
                   3970:       (void) add_local_symbol ((const char *)0, (const char *)0,
                   3971:                               st_End, sc_Text,
                   3972:                               (symint_t)0,
                   3973:                               (symint_t)0);
                   3974: 
                   3975:       file_ptr->fdr.cpd = file_ptr->procs.num_allocated;
                   3976:       file_ptr->fdr.ipdFirst = symbolic_header.ipdMax;
                   3977:       symbolic_header.ipdMax += file_ptr->fdr.cpd;
                   3978: 
                   3979:       file_ptr->fdr.csym = file_ptr->symbols.num_allocated;
                   3980:       file_ptr->fdr.isymBase = symbolic_header.isymMax;
                   3981:       symbolic_header.isymMax += file_ptr->fdr.csym;
                   3982: 
                   3983:       file_ptr->fdr.caux = file_ptr->aux_syms.num_allocated;
                   3984:       file_ptr->fdr.iauxBase = symbolic_header.iauxMax;
                   3985:       symbolic_header.iauxMax += file_ptr->fdr.caux;
                   3986: 
                   3987:       file_ptr->fdr.cbSs = file_ptr->strings.num_allocated;
                   3988:       file_ptr->fdr.issBase = symbolic_header.issMax;
                   3989:       symbolic_header.issMax += file_ptr->fdr.cbSs;
                   3990:     }
                   3991: 
                   3992: 
                   3993:   i = WORD_ALIGN (symbolic_header.cbLine);     /* line numbers */
                   3994:   if (i > 0)
                   3995:     {
                   3996:       symbolic_header.cbLineOffset = file_offset;
                   3997:       file_offset += i;
                   3998:     }
                   3999: 
                   4000:   i = symbolic_header.ioptMax;                 /* optimization symbols */
                   4001:   if (((long) i) > 0)
                   4002:     {
                   4003:       symbolic_header.cbOptOffset = file_offset;
                   4004:       file_offset += i * sizeof (OPTR);
                   4005:     }
                   4006: 
                   4007:   i = symbolic_header.idnMax;                  /* dense numbers */
                   4008:   if (i > 0)
                   4009:     {
                   4010:       symbolic_header.cbDnOffset = file_offset;
                   4011:       file_offset += i * sizeof (DNR);
                   4012:     }
                   4013: 
                   4014:   i = symbolic_header.ipdMax;                  /* procedure tables */
                   4015:   if (i > 0)
                   4016:     {
                   4017:       symbolic_header.cbPdOffset = file_offset;
                   4018:       file_offset += i * sizeof (PDR);
                   4019:     }
                   4020: 
                   4021:   i = symbolic_header.isymMax;                 /* local symbols */
                   4022:   if (i > 0)
                   4023:     {
                   4024:       symbolic_header.cbSymOffset = file_offset;
                   4025:       file_offset += i * sizeof (SYMR);
                   4026:     }
                   4027: 
                   4028:   i = symbolic_header.iauxMax;                 /* aux syms. */
                   4029:   if (i > 0)
                   4030:     {
                   4031:       symbolic_header.cbAuxOffset = file_offset;
                   4032:       file_offset += i * sizeof (TIR);
                   4033:     }
                   4034: 
                   4035:   i = WORD_ALIGN (symbolic_header.issMax);     /* local strings */
                   4036:   if (i > 0)
                   4037:     {
                   4038:       symbolic_header.cbSsOffset = file_offset;
                   4039:       file_offset += i;
                   4040:     }
                   4041: 
                   4042:   i = WORD_ALIGN (symbolic_header.issExtMax);  /* external strings */
                   4043:   if (i > 0)
                   4044:     {
                   4045:       symbolic_header.cbSsExtOffset = file_offset;
                   4046:       file_offset += i;
                   4047:     }
                   4048: 
                   4049:   i = symbolic_header.ifdMax;                  /* file tables */
                   4050:   if (i > 0)
                   4051:     {
                   4052:       symbolic_header.cbFdOffset = file_offset;
                   4053:       file_offset += i * sizeof (FDR);
                   4054:     }
                   4055: 
                   4056:   i = symbolic_header.crfd;                    /* relative file descriptors */
                   4057:   if (i > 0)
                   4058:     {
                   4059:       symbolic_header.cbRfdOffset = file_offset;
                   4060:       file_offset += i * sizeof (symint_t);
                   4061:     }
                   4062: 
                   4063:   i = symbolic_header.iextMax;                 /* external symbols */
                   4064:   if (i > 0)
                   4065:     {
                   4066:       symbolic_header.cbExtOffset = file_offset;
                   4067:       file_offset += i * sizeof (EXTR);
                   4068:     }
                   4069: }
                   4070: 
                   4071: 
                   4072: /* Write out a varray at a given location.  */
                   4073: 
                   4074: STATIC void
                   4075: write_varray (vp, offset, str)
                   4076:      varray_t *vp;                     /* virtual array */
                   4077:      off_t offset;                     /* offset to write varray to */
                   4078:      const char *str;                  /* string to print out when tracing */
                   4079: {
                   4080:   int num_write, sys_write;
                   4081:   vlinks_t *ptr;
                   4082: 
                   4083:   if (vp->num_allocated == 0)
                   4084:     return;
                   4085: 
                   4086:   if (debug)
                   4087:     fprintf (stderr, "\twarray\tvp = 0x%.8x, offset = %7u, size = %7u, %s\n",
                   4088:             vp, offset, vp->num_allocated * vp->object_size, str);
                   4089: 
                   4090:   if (file_offset != offset
                   4091:       && fseek (object_stream, (long)offset, SEEK_SET) < 0)
                   4092:     pfatal_with_name (object_name);
                   4093: 
                   4094:   for (ptr = vp->first; ptr != (vlinks_t *)0; ptr = ptr->next)
                   4095:     {
                   4096:       num_write = (ptr->next == (vlinks_t *)0)
                   4097:        ? vp->objects_last_page * vp->object_size
                   4098:        : vp->objects_per_page  * vp->object_size;
                   4099: 
                   4100:       sys_write = fwrite ((PTR_T) ptr->datum, 1, num_write, object_stream);
                   4101:       if (sys_write <= 0)
                   4102:        pfatal_with_name (object_name);
                   4103: 
                   4104:       else if (sys_write != num_write)
                   4105:        fatal ("Wrote %d bytes to %s, system returned %d",
                   4106:               num_write,
                   4107:               object_name,
                   4108:               sys_write);
                   4109: 
                   4110:       file_offset += num_write;
                   4111:     }
                   4112: }
                   4113: 
                   4114: 
                   4115: /* Write out the symbol table in the object file.  */
                   4116: 
                   4117: STATIC void
                   4118: write_object __proto((void))
                   4119: {
                   4120:   int sys_write;
                   4121:   efdr_t *file_ptr;
                   4122:   off_t offset;
                   4123: 
                   4124:   if (debug)
                   4125:     fprintf (stderr, "\n\twrite\tvp = 0x%.8x, offset = %7u, size = %7u, %s\n",
                   4126:             (PTR_T *) &symbolic_header, 0, sizeof (symbolic_header),
                   4127:             "symbolic header");
                   4128: 
                   4129:   sys_write = fwrite ((PTR_T) &symbolic_header,
                   4130:                      1,
                   4131:                      sizeof (symbolic_header),
                   4132:                      object_stream);
                   4133: 
                   4134:   if (sys_write < 0)
                   4135:     pfatal_with_name (object_name);
                   4136: 
                   4137:   else if (sys_write != sizeof (symbolic_header))
                   4138:     fatal ("Wrote %d bytes to %s, system returned %d",
                   4139:           sizeof (symbolic_header),
                   4140:           object_name,
                   4141:           sys_write);
                   4142: 
                   4143: 
                   4144:   file_offset = sizeof (symbolic_header) + orig_file_header.f_symptr;
                   4145: 
                   4146:   if (symbolic_header.cbLine > 0)              /* line numbers */
                   4147:     {
                   4148:       long sys_write;
                   4149: 
                   4150:       if (file_offset != symbolic_header.cbLineOffset
                   4151:          && fseek (object_stream, symbolic_header.cbLineOffset, SEEK_SET) != 0)
                   4152:        pfatal_with_name (object_name);
                   4153: 
                   4154:       if (debug)
                   4155:        fprintf (stderr, "\twrite\tvp = 0x%.8x, offset = %7u, size = %7u, %s\n",
                   4156:                 (PTR_T *) &orig_linenum, symbolic_header.cbLineOffset,
                   4157:                 symbolic_header.cbLine, "Line numbers");
                   4158: 
                   4159:       sys_write = fwrite ((PTR_T) orig_linenum,
                   4160:                          1,
                   4161:                          symbolic_header.cbLine,
                   4162:                          object_stream);
                   4163: 
                   4164:       if (sys_write <= 0)
                   4165:        pfatal_with_name (object_name);
                   4166: 
                   4167:       else if (sys_write != symbolic_header.cbLine)
                   4168:        fatal ("Wrote %d bytes to %s, system returned %d",
                   4169:               symbolic_header.cbLine,
                   4170:               object_name,
                   4171:               sys_write);
                   4172: 
                   4173:       file_offset = symbolic_header.cbLineOffset + symbolic_header.cbLine;
                   4174:     }
                   4175: 
                   4176:   if (symbolic_header.ioptMax > 0)             /* optimization symbols */
                   4177:     {
                   4178:       long sys_write;
                   4179:       long num_write = symbolic_header.ioptMax * sizeof (OPTR);
                   4180: 
                   4181:       if (file_offset != symbolic_header.cbOptOffset
                   4182:          && fseek (object_stream, symbolic_header.cbOptOffset, SEEK_SET) != 0)
                   4183:        pfatal_with_name (object_name);
                   4184: 
                   4185:       if (debug)
                   4186:        fprintf (stderr, "\twrite\tvp = 0x%.8x, offset = %7u, size = %7u, %s\n",
                   4187:                 (PTR_T *) &orig_opt_syms, symbolic_header.cbOptOffset,
                   4188:                 num_write, "Optimizer symbols");
                   4189: 
                   4190:       sys_write = fwrite ((PTR_T) orig_opt_syms,
                   4191:                          1,
                   4192:                          num_write,
                   4193:                          object_stream);
                   4194: 
                   4195:       if (sys_write <= 0)
                   4196:        pfatal_with_name (object_name);
                   4197: 
                   4198:       else if (sys_write != num_write)
                   4199:        fatal ("Wrote %d bytes to %s, system returned %d",
                   4200:               num_write,
                   4201:               object_name,
                   4202:               sys_write);
                   4203: 
                   4204:       file_offset = symbolic_header.cbOptOffset + num_write;
                   4205:     }
                   4206: 
                   4207:   if (symbolic_header.idnMax > 0)              /* dense numbers */
                   4208:     write_varray (&dense_num, (off_t)symbolic_header.cbDnOffset, "Dense numbers");
                   4209: 
                   4210:   if (symbolic_header.ipdMax > 0)              /* procedure tables */
                   4211:     {
                   4212:       offset = symbolic_header.cbPdOffset;
                   4213:       for (file_ptr = first_file;
                   4214:           file_ptr != (efdr_t *)0;
                   4215:           file_ptr = file_ptr->next_file)
                   4216:        {
                   4217:          write_varray (&file_ptr->procs, offset, "Procedure tables");
                   4218:          offset = file_offset;
                   4219:        }
                   4220:     }
                   4221: 
                   4222:   if (symbolic_header.isymMax > 0)             /* local symbols */
                   4223:     {
                   4224:       offset = symbolic_header.cbSymOffset;
                   4225:       for (file_ptr = first_file;
                   4226:           file_ptr != (efdr_t *)0;
                   4227:           file_ptr = file_ptr->next_file)
                   4228:        {
                   4229:          write_varray (&file_ptr->symbols, offset, "Local symbols");
                   4230:          offset = file_offset;
                   4231:        }
                   4232:     }
                   4233: 
                   4234:   if (symbolic_header.iauxMax > 0)             /* aux symbols */
                   4235:     {
                   4236:       offset = symbolic_header.cbAuxOffset;
                   4237:       for (file_ptr = first_file;
                   4238:           file_ptr != (efdr_t *)0;
                   4239:           file_ptr = file_ptr->next_file)
                   4240:        {
                   4241:          write_varray (&file_ptr->aux_syms, offset, "Aux. symbols");
                   4242:          offset = file_offset;
                   4243:        }
                   4244:     }
                   4245: 
                   4246:   if (symbolic_header.issMax > 0)              /* local strings */
                   4247:     {
                   4248:       offset = symbolic_header.cbSsOffset;
                   4249:       for (file_ptr = first_file;
                   4250:           file_ptr != (efdr_t *)0;
                   4251:           file_ptr = file_ptr->next_file)
                   4252:        {
                   4253:          write_varray (&file_ptr->strings, offset, "Local strings");
                   4254:          offset = file_offset;
                   4255:        }
                   4256:     }
                   4257: 
                   4258:   if (symbolic_header.issExtMax > 0)           /* external strings */
                   4259:     write_varray (&ext_strings, symbolic_header.cbSsExtOffset, "External strings");
                   4260: 
                   4261:   if (symbolic_header.ifdMax > 0)              /* file tables */
                   4262:     {
                   4263:       offset = symbolic_header.cbFdOffset;
                   4264:       if (file_offset != offset
                   4265:          && fseek (object_stream, (long)offset, SEEK_SET) < 0)
                   4266:        pfatal_with_name (object_name);
                   4267: 
                   4268:       file_offset = offset;
                   4269:       for (file_ptr = first_file;
                   4270:           file_ptr != (efdr_t *)0;
                   4271:           file_ptr = file_ptr->next_file)
                   4272:        {
                   4273:          if (debug)
                   4274:            fprintf (stderr, "\twrite\tvp = 0x%.8x, offset = %7u, size = %7u, %s\n",
                   4275:                     (PTR_T *) &file_ptr->fdr, file_offset, sizeof (FDR), "File header");
                   4276: 
                   4277:          sys_write = fwrite (&file_ptr->fdr,
                   4278:                              1,
                   4279:                              sizeof (FDR),
                   4280:                              object_stream);
                   4281: 
                   4282:          if (sys_write < 0)
                   4283:            pfatal_with_name (object_name);
                   4284: 
                   4285:          else if (sys_write != sizeof (FDR))
                   4286:            fatal ("Wrote %d bytes to %s, system returned %d",
                   4287:                   sizeof (FDR),
                   4288:                   object_name,
                   4289:                   sys_write);
                   4290: 
                   4291:          file_offset = offset += sizeof (FDR);
                   4292:        }
                   4293:     }
                   4294: 
                   4295:   if (symbolic_header.crfd > 0)                        /* relative file descriptors */
                   4296:     {
                   4297:       long sys_write;
                   4298:       symint_t num_write = symbolic_header.crfd * sizeof (symint_t);
                   4299: 
                   4300:       if (file_offset != symbolic_header.cbRfdOffset
                   4301:          && fseek (object_stream, symbolic_header.cbRfdOffset, SEEK_SET) != 0)
                   4302:        pfatal_with_name (object_name);
                   4303: 
                   4304:       if (debug)
                   4305:        fprintf (stderr, "\twrite\tvp = 0x%.8x, offset = %7u, size = %7u, %s\n",
                   4306:                 (PTR_T *) &orig_rfds, symbolic_header.cbRfdOffset,
                   4307:                 num_write, "Relative file descriptors");
                   4308: 
                   4309:       sys_write = fwrite (orig_rfds,
                   4310:                          1,
                   4311:                          num_write,
                   4312:                          object_stream);
                   4313: 
                   4314:       if (sys_write <= 0)
                   4315:        pfatal_with_name (object_name);
                   4316: 
                   4317:       else if (sys_write != num_write)
                   4318:        fatal ("Wrote %d bytes to %s, system returned %d",
                   4319:               num_write,
                   4320:               object_name,
                   4321:               sys_write);
                   4322: 
                   4323:       file_offset = symbolic_header.cbRfdOffset + num_write;
                   4324:     }
                   4325: 
                   4326:   if (symbolic_header.issExtMax > 0)           /* external symbols */
                   4327:     write_varray (&ext_symbols, (off_t)symbolic_header.cbExtOffset, "External symbols");
                   4328: 
                   4329:   if (fclose (object_stream) != 0)
                   4330:     pfatal_with_name (object_name);
                   4331: }
                   4332: 
                   4333: 
                   4334: /* Read some bytes at a specified location, and return a pointer.  */
                   4335: 
                   4336: STATIC page_t *
                   4337: read_seek (size, offset, str)
                   4338:      Size_t size;              /* # bytes to read */
                   4339:      off_t offset;             /* offset to read at */
                   4340:      const char *str;          /* name for tracing */
                   4341: {
                   4342:   page_t *ptr;
                   4343:   long sys_read = 0;
                   4344: 
                   4345:   if (size == 0)               /* nothing to read */
                   4346:     return (page_t *)0;
                   4347: 
                   4348:   if (debug)
                   4349:     fprintf (stderr, "\trseek\tsize = %7u, offset = %7u, currently at %7u, %s\n",
                   4350:             size, offset, file_offset, str);
                   4351: 
                   4352: #ifndef MALLOC_CHECK
                   4353:   ptr = allocate_multiple_pages ((size + PAGE_USIZE - 1) / PAGE_USIZE);
                   4354: #else
                   4355:   ptr = (page_t *) xcalloc (1, size);
                   4356: #endif
                   4357: 
                   4358:   /* If we need to seek, and the distance is nearby, just do some reads,
                   4359:      to speed things up.  */
                   4360:   if (file_offset != offset)
                   4361:     {
                   4362:       symint_t difference = offset - file_offset;
                   4363: 
                   4364:       if (difference < 8)
                   4365:        {
                   4366:          char small_buffer[8];
                   4367: 
                   4368:          sys_read = fread (small_buffer, 1, difference, obj_in_stream);
                   4369:          if (sys_read <= 0)
                   4370:            pfatal_with_name (obj_in_name);
                   4371: 
                   4372:          if (sys_read != difference)
                   4373:            fatal ("Wanted to read %d bytes from %s, system returned %d",
                   4374:                   size,
                   4375:                   obj_in_name,
                   4376:                   sys_read);
                   4377:        }
                   4378:       else if (fseek (obj_in_stream, offset, SEEK_SET) < 0)
                   4379:        pfatal_with_name (obj_in_name);
                   4380:     }
                   4381: 
                   4382:   sys_read = fread ((PTR_T)ptr, 1, size, obj_in_stream);
                   4383:   if (sys_read <= 0)
                   4384:     pfatal_with_name (obj_in_name);
                   4385: 
                   4386:   if (sys_read != size)
                   4387:     fatal ("Wanted to read %d bytes from %s, system returned %d",
                   4388:           size,
                   4389:           obj_in_name,
                   4390:           sys_read);
                   4391: 
                   4392:   file_offset = offset + size;
                   4393: 
                   4394:   if (file_offset > max_file_offset)
                   4395:     max_file_offset = file_offset;
                   4396: 
                   4397:   return ptr;
                   4398: }
                   4399: 
                   4400: 
                   4401: /* Read the existing object file (and copy to the output object file
                   4402:    if it is different from the input object file), and remove the old
                   4403:    symbol table.  */
                   4404: 
                   4405: STATIC void
                   4406: copy_object __proto((void))
                   4407: {
                   4408:   char buffer[ PAGE_SIZE ];
                   4409:   register int sys_read;
                   4410:   register int remaining;
                   4411:   register int num_write;
                   4412:   register int sys_write;
                   4413:   register int fd, es;
                   4414:   register int delete_ifd = 0;
                   4415:   register int *remap_file_number;
                   4416:   struct stat stat_buf;
                   4417: 
                   4418:   if (debug)
                   4419:     fprintf (stderr, "\tcopy\n");
                   4420: 
                   4421:   if (fstat (fileno (obj_in_stream), &stat_buf) != 0
                   4422:       || fseek (obj_in_stream, 0L, SEEK_SET) != 0)
                   4423:     pfatal_with_name (obj_in_name);
                   4424: 
                   4425:   sys_read = fread ((PTR_T) &orig_file_header,
                   4426:                    1,
                   4427:                    sizeof (struct filehdr),
                   4428:                    obj_in_stream);
                   4429: 
                   4430:   if (sys_read < 0)
                   4431:     pfatal_with_name (obj_in_name);
                   4432: 
                   4433:   else if (sys_read == 0 && feof (obj_in_stream))
                   4434:     return;                    /* create a .T file sans file header */
                   4435: 
                   4436:   else if (sys_read < sizeof (struct filehdr))
                   4437:     fatal ("Wanted to read %d bytes from %s, system returned %d",
                   4438:           sizeof (struct filehdr),
                   4439:           obj_in_name,
                   4440:           sys_read);
                   4441: 
                   4442: 
                   4443:   if (orig_file_header.f_nsyms != sizeof (HDRR))
                   4444:     fatal ("%s symbolic header wrong size (%d bytes, should be %d)",
                   4445:           input_name, orig_file_header.f_nsyms, sizeof (HDRR));
                   4446: 
                   4447: 
                   4448:   /* Read in the current symbolic header.  */
                   4449:   if (fseek (obj_in_stream, (long) orig_file_header.f_symptr, SEEK_SET) != 0)
                   4450:     pfatal_with_name (input_name);
                   4451: 
                   4452:   sys_read = fread ((PTR_T) &orig_sym_hdr,
                   4453:                    1,
                   4454:                    sizeof (orig_sym_hdr),
                   4455:                    obj_in_stream);
                   4456: 
                   4457:   if (sys_read < 0)
                   4458:     pfatal_with_name (object_name);
                   4459: 
                   4460:   else if (sys_read < sizeof (struct filehdr))
                   4461:     fatal ("Wanted to read %d bytes from %s, system returned %d",
                   4462:           sizeof (struct filehdr),
                   4463:           obj_in_name,
                   4464:           sys_read);
                   4465: 
                   4466: 
                   4467:   /* Read in each of the sections if they exist in the object file.
                   4468:      We read things in in the order the mips assembler creates the
                   4469:      sections, so in theory no extra seeks are done.
                   4470: 
                   4471:      For simplicity sake, round each read up to a page boundary,
                   4472:      we may want to revisit this later.... */
                   4473: 
                   4474:   file_offset =  orig_file_header.f_symptr + sizeof (struct filehdr);
                   4475: 
                   4476:   if (orig_sym_hdr.cbLine > 0)                 /* line numbers */
                   4477:     orig_linenum = (char *) read_seek ((Size_t)orig_sym_hdr.cbLine,
                   4478:                                       orig_sym_hdr.cbLineOffset,
                   4479:                                       "Line numbers");
                   4480: 
                   4481:   if (orig_sym_hdr.ipdMax > 0)                 /* procedure tables */
                   4482:     orig_procs = (PDR *) read_seek ((Size_t)orig_sym_hdr.ipdMax * sizeof (PDR),
                   4483:                                    orig_sym_hdr.cbPdOffset,
                   4484:                                    "Procedure tables");
                   4485: 
                   4486:   if (orig_sym_hdr.isymMax > 0)                        /* local symbols */
                   4487:     orig_local_syms = (SYMR *) read_seek ((Size_t)orig_sym_hdr.isymMax * sizeof (SYMR),
                   4488:                                          orig_sym_hdr.cbSymOffset,
                   4489:                                          "Local symbols");
                   4490: 
                   4491:   if (orig_sym_hdr.iauxMax > 0)                        /* aux symbols */
                   4492:     orig_aux_syms = (AUXU *) read_seek ((Size_t)orig_sym_hdr.iauxMax * sizeof (AUXU),
                   4493:                                        orig_sym_hdr.cbAuxOffset,
                   4494:                                        "Aux. symbols");
                   4495: 
                   4496:   if (orig_sym_hdr.issMax > 0)                 /* local strings */
                   4497:     orig_local_strs = (char *) read_seek ((Size_t)orig_sym_hdr.issMax,
                   4498:                                          orig_sym_hdr.cbSsOffset,
                   4499:                                          "Local strings");
                   4500: 
                   4501:   if (orig_sym_hdr.issExtMax > 0)              /* external strings */
                   4502:     orig_ext_strs = (char *) read_seek ((Size_t)orig_sym_hdr.issExtMax,
                   4503:                                        orig_sym_hdr.cbSsExtOffset,
                   4504:                                        "External strings");
                   4505: 
                   4506:   if (orig_sym_hdr.ifdMax > 0)                 /* file tables */
                   4507:     orig_files = (FDR *) read_seek ((Size_t)orig_sym_hdr.ifdMax * sizeof (FDR),
                   4508:                                    orig_sym_hdr.cbFdOffset,
                   4509:                                    "File tables");
                   4510: 
                   4511:   if (orig_sym_hdr.crfd > 0)                   /* relative file descriptors */
                   4512:     orig_rfds = (symint_t *) read_seek ((Size_t)orig_sym_hdr.crfd * sizeof (symint_t),
                   4513:                                        orig_sym_hdr.cbRfdOffset,
                   4514:                                        "Relative file descriptors");
                   4515: 
                   4516:   if (orig_sym_hdr.issExtMax > 0)              /* external symbols */
                   4517:     orig_ext_syms = (EXTR *) read_seek ((Size_t)orig_sym_hdr.iextMax * sizeof (EXTR),
                   4518:                                        orig_sym_hdr.cbExtOffset,
                   4519:                                        "External symbols");
                   4520: 
                   4521:   if (orig_sym_hdr.idnMax > 0)                 /* dense numbers */
                   4522:     {
                   4523:       orig_dense = (DNR *) read_seek ((Size_t)orig_sym_hdr.idnMax * sizeof (DNR),
                   4524:                                      orig_sym_hdr.cbDnOffset,
                   4525:                                      "Dense numbers");
                   4526: 
                   4527:       add_bytes (&dense_num, (char *) orig_dense, (Size_t)orig_sym_hdr.idnMax);
                   4528:     }
                   4529: 
                   4530:   if (orig_sym_hdr.ioptMax > 0)                        /* opt symbols */
                   4531:     orig_opt_syms = (OPTR *) read_seek ((Size_t)orig_sym_hdr.ioptMax * sizeof (OPTR),
                   4532:                                        orig_sym_hdr.cbOptOffset,
                   4533:                                        "Optimizer symbols");
                   4534: 
                   4535: 
                   4536: 
                   4537:   /* Abort if the symbol table is not last.  */
                   4538:   if (max_file_offset != stat_buf.st_size)
                   4539:     fatal ("Symbol table is not last (symbol table ends at %ld, .o ends at %ld",
                   4540:           max_file_offset,
                   4541:           stat_buf.st_size);
                   4542: 
                   4543: 
                   4544:   /* If the first original file descriptor is a dummy which the assembler
                   4545:      put out, but there are no symbols in it, skip it now.  */
                   4546:   if (orig_sym_hdr.ifdMax > 1
                   4547:       && orig_files->csym == 2
                   4548:       && orig_files->caux == 0)
                   4549:     {
                   4550:       char *filename = orig_local_strs + (orig_files->issBase + orig_files->rss);
                   4551:       char *suffix = local_rindex (filename, '.');
                   4552: 
                   4553:       if (suffix != (char *)0 && strcmp (suffix, ".s") == 0)
                   4554:        delete_ifd = 1;
                   4555:     }
                   4556: 
                   4557: 
                   4558:   /* Create array to map original file numbers to the new file numbers
                   4559:      (in case there are duplicate filenames, we collapse them into one
                   4560:      file section, the MIPS assembler may or may not collapse them).  */
                   4561: 
                   4562:   remap_file_number = (int *) alloca (sizeof (int) * orig_sym_hdr.ifdMax);
                   4563: 
                   4564:   for (fd = delete_ifd; fd < orig_sym_hdr.ifdMax; fd++)
                   4565:     {
                   4566:       register FDR *fd_ptr = ORIG_FILES (fd);
                   4567:       register char *filename = ORIG_LSTRS (fd_ptr->issBase + fd_ptr->rss);
                   4568: 
                   4569:       /* file support itself.  */
                   4570:       add_file (filename, filename + strlen (filename));
                   4571:       remap_file_number[fd] = cur_file_ptr->file_index;
                   4572:     }
                   4573: 
                   4574:   if (delete_ifd > 0)          /* just in case */
                   4575:     remap_file_number[0] = remap_file_number[1];
                   4576: 
                   4577: 
                   4578:   /* Loop, adding each of the external symbols.  These must be in
                   4579:      order or otherwise we would have to change the relocation
                   4580:      entries.  We don't just call add_bytes, because we need to have
                   4581:      the names put into the external hash table.  We set the type to
                   4582:      'void' for now, and parse_def will fill in the correct type if it
                   4583:      is in the symbol table.  We must add the external symbols before
                   4584:      the locals, since the locals do lookups against the externals.  */
                   4585: 
                   4586:   if (debug)
                   4587:     fprintf (stderr, "\tehash\n");
                   4588: 
                   4589:   for (es = 0; es < orig_sym_hdr.iextMax; es++)
                   4590:     {
                   4591:       register EXTR *eptr = orig_ext_syms + es;
                   4592:       register char *ename = ORIG_ESTRS (eptr->asym.iss);
                   4593:       register unsigned ifd = eptr->ifd;
                   4594: 
                   4595:       (void) add_ext_symbol (ename,
                   4596:                             ename + strlen (ename),
                   4597:                             (st_t) eptr->asym.st,
                   4598:                             (sc_t) eptr->asym.sc,
                   4599:                             eptr->asym.value,
                   4600:                             (symint_t)((eptr->asym.index == indexNil) ? indexNil : 0),
                   4601:                             (ifd < orig_sym_hdr.ifdMax) ? remap_file_number[ ifd ] : ifd);
                   4602:     }
                   4603: 
                   4604: 
                   4605:   /* For each of the files in the object file, copy the symbols, and such
                   4606:      into the varrays for the new object file.  */
                   4607: 
                   4608:   for (fd = delete_ifd; fd < orig_sym_hdr.ifdMax; fd++)
                   4609:     {
                   4610:       register FDR *fd_ptr = ORIG_FILES (fd);
                   4611:       register char *filename = ORIG_LSTRS (fd_ptr->issBase + fd_ptr->rss);
                   4612:       register SYMR *sym_start;
                   4613:       register SYMR *sym;
                   4614:       register SYMR *sym_end_p1;
                   4615:       register PDR *proc_start;
                   4616:       register PDR *proc;
                   4617:       register PDR *proc_end_p1;
                   4618: 
                   4619:       /* file support itself.  */
                   4620:       add_file (filename, filename + strlen (filename));
                   4621:       cur_file_ptr->orig_fdr = fd_ptr;
                   4622: 
                   4623:       /* Copy stuff that's just passed through (such as line #'s) */
                   4624:       cur_file_ptr->fdr.adr         = fd_ptr->adr;
                   4625:       cur_file_ptr->fdr.ilineBase    = fd_ptr->ilineBase;
                   4626:       cur_file_ptr->fdr.cline       = fd_ptr->cline;
                   4627:       cur_file_ptr->fdr.rfdBase             = fd_ptr->rfdBase;
                   4628:       cur_file_ptr->fdr.crfd        = fd_ptr->crfd;
                   4629:       cur_file_ptr->fdr.cbLineOffset = fd_ptr->cbLineOffset;
                   4630:       cur_file_ptr->fdr.cbLine      = fd_ptr->cbLine;
                   4631:       cur_file_ptr->fdr.fMerge      = fd_ptr->fMerge;
                   4632:       cur_file_ptr->fdr.fReadin             = fd_ptr->fReadin;
                   4633:       cur_file_ptr->fdr.glevel      = fd_ptr->glevel;
                   4634: 
                   4635:       if (debug)
                   4636:        fprintf (stderr, "\thash\tstart, filename %s\n", filename);
                   4637: 
                   4638:       /* For each of the static and global symbols defined, add them
                   4639:         to the hash table of original symbols, so we can look up
                   4640:         their values.  */
                   4641: 
                   4642:       sym_start = ORIG_LSYMS (fd_ptr->isymBase);
                   4643:       sym_end_p1 = sym_start + fd_ptr->csym;
                   4644:       for (sym = sym_start; sym < sym_end_p1; sym++)
                   4645:        {
                   4646:          switch ((st_t) sym->st)
                   4647:            {
                   4648:            default:
                   4649:              break;
                   4650: 
                   4651:            case st_Global:
                   4652:            case st_Static:
                   4653:            case st_Label:
                   4654:            case st_Proc:
                   4655:            case st_StaticProc:
                   4656:              {
                   4657:                auto symint_t hash_index;
                   4658:                register char *str = ORIG_LSTRS (fd_ptr->issBase + sym->iss);
                   4659:                register Size_t len = strlen (str);
                   4660:                register shash_t *shash_ptr = hash_string (str,
                   4661:                                                           (Ptrdiff_t)len,
                   4662:                                                           &orig_str_hash[0],
                   4663:                                                           &hash_index);
                   4664: 
                   4665:                if (shash_ptr != (shash_t *)0)
                   4666:                  error ("internal error, %s is already in original symbol table", str);
                   4667: 
                   4668:                else
                   4669:                  {
                   4670:                    shash_ptr = allocate_shash ();
                   4671:                    shash_ptr->next = orig_str_hash[hash_index];
                   4672:                    orig_str_hash[hash_index] = shash_ptr;
                   4673: 
                   4674:                    shash_ptr->len = len;
                   4675:                    shash_ptr->indx = indexNil;
                   4676:                    shash_ptr->string = str;
                   4677:                    shash_ptr->sym_ptr = sym;
                   4678:                  }
                   4679:              }
                   4680:              break;
                   4681: 
                   4682:            case st_End:
                   4683:              if ((sc_t) sym->sc == sc_Text)
                   4684:                {
                   4685:                  register char *str = ORIG_LSTRS (fd_ptr->issBase + sym->iss);
                   4686: 
                   4687:                  if (*str != '\0')
                   4688:                    {
                   4689:                      register Size_t len = strlen (str);
                   4690:                      register shash_t *shash_ptr = hash_string (str,
                   4691:                                                                 (Ptrdiff_t)len,
                   4692:                                                                 &orig_str_hash[0],
                   4693:                                                                 (symint_t *)0);
                   4694: 
                   4695:                      if (shash_ptr != (shash_t *)0)
                   4696:                        shash_ptr->end_ptr = sym;
                   4697:                    }
                   4698:                }
                   4699:              break;
                   4700: 
                   4701:            }
                   4702:        }
                   4703: 
                   4704:       if (debug)
                   4705:        {
                   4706:          fprintf (stderr, "\thash\tdone,  filename %s\n", filename);
                   4707:          fprintf (stderr, "\tproc\tstart, filename %s\n", filename);
                   4708:        }
                   4709: 
                   4710:       /* Go through each of the procedures in this file, and add the
                   4711:         procedure pointer to the hash entry for the given name.  */
                   4712: 
                   4713:       proc_start = ORIG_PROCS (fd_ptr->ipdFirst);
                   4714:       proc_end_p1 = proc_start + fd_ptr->cpd;
                   4715:       for (proc = proc_start; proc < proc_end_p1; proc++)
                   4716:        {
                   4717:          register SYMR *proc_sym = ORIG_LSYMS (fd_ptr->isymBase + proc->isym);
                   4718:          register char *str = ORIG_LSTRS (fd_ptr->issBase + proc_sym->iss);
                   4719:          register Size_t len = strlen (str);
                   4720:          register shash_t *shash_ptr = hash_string (str,
                   4721:                                                     (Ptrdiff_t)len,
                   4722:                                                     &orig_str_hash[0],
                   4723:                                                     (symint_t *)0);
                   4724: 
                   4725:          if (shash_ptr == (shash_t *)0)
                   4726:            error ("internal error, function %s is not in original symbol table", str);
                   4727: 
                   4728:          else
                   4729:            shash_ptr->proc_ptr = proc;
                   4730:        }
                   4731: 
                   4732:       if (debug)
                   4733:        fprintf (stderr, "\tproc\tdone,  filename %s\n", filename);
                   4734: 
                   4735:     }
                   4736:   cur_file_ptr = first_file;
                   4737: 
                   4738: 
                   4739:   /* Copy all of the object file up to the symbol table.  Originally
                   4740:      we were going to use ftruncate, but that doesn't seem to work
                   4741:      on Ultrix 3.1.... */
                   4742: 
                   4743:   if (fseek (obj_in_stream, (long)0, SEEK_SET) != 0)
                   4744:     pfatal_with_name (obj_in_name);
                   4745: 
                   4746:   if (fseek (object_stream, (long)0, SEEK_SET) != 0)
                   4747:     pfatal_with_name (object_name);
                   4748: 
                   4749:   for (remaining = orig_file_header.f_symptr;
                   4750:        remaining > 0;
                   4751:        remaining -= num_write)
                   4752:     {
                   4753:       num_write = (remaining <= sizeof (buffer)) ? remaining : sizeof (buffer);
                   4754:       sys_read = fread ((PTR_T) buffer, 1, num_write, obj_in_stream);
                   4755:       if (sys_read <= 0)
                   4756:        pfatal_with_name (obj_in_name);
                   4757: 
                   4758:       else if (sys_read != num_write)
                   4759:        fatal ("Wanted to read %d bytes from %s, system returned %d",
                   4760:               num_write,
                   4761:               obj_in_name,
                   4762:               sys_read);
                   4763: 
                   4764:       sys_write = fwrite (buffer, 1, num_write, object_stream);
                   4765:       if (sys_write <= 0)
                   4766:        pfatal_with_name (object_name);
                   4767: 
                   4768:       else if (sys_write != num_write)
                   4769:        fatal ("Wrote %d bytes to %s, system returned %d",
                   4770:               num_write,
                   4771:               object_name,
                   4772:               sys_write);
                   4773:     }
                   4774: }
                   4775: 
                   4776: 
                   4777: /* Ye olde main program.  */
                   4778: 
                   4779: int
                   4780: main (argc, argv)
                   4781:      int argc;
                   4782:      char *argv[];
                   4783: {
                   4784:   int iflag = 0;
                   4785:   char *p = local_rindex (argv[0], '/');
                   4786:   char *num_end;
                   4787:   int option;
                   4788:   int i;
                   4789: 
                   4790:   progname = (p != 0) ? p+1 : argv[0];
                   4791: 
                   4792:   (void) signal (SIGSEGV, catch_signal);
                   4793:   (void) signal (SIGBUS,  catch_signal);
                   4794:   (void) signal (SIGABRT, catch_signal);
                   4795: 
                   4796: #if !defined(__SABER__) && !defined(lint)
                   4797:   if (sizeof (efdr_t) > PAGE_USIZE)
                   4798:     fatal ("Efdr_t has a sizeof %d bytes, when it should be less than %d",
                   4799:           sizeof (efdr_t),
                   4800:           PAGE_USIZE);
                   4801: 
                   4802:   if (sizeof (page_t) != PAGE_USIZE)
                   4803:     fatal ("Page_t has a sizeof %d bytes, when it should be %d",
                   4804:           sizeof (page_t),
                   4805:           PAGE_USIZE);
                   4806: 
                   4807: #endif
                   4808: 
                   4809:   alloc_counts[ alloc_type_none    ].alloc_name = "none";
                   4810:   alloc_counts[ alloc_type_scope   ].alloc_name = "scope";
                   4811:   alloc_counts[ alloc_type_vlinks  ].alloc_name = "vlinks";
                   4812:   alloc_counts[ alloc_type_shash   ].alloc_name = "shash";
                   4813:   alloc_counts[ alloc_type_thash   ].alloc_name = "thash";
                   4814:   alloc_counts[ alloc_type_tag     ].alloc_name = "tag";
                   4815:   alloc_counts[ alloc_type_forward ].alloc_name = "forward";
                   4816:   alloc_counts[ alloc_type_thead   ].alloc_name = "thead";
                   4817:   alloc_counts[ alloc_type_varray  ].alloc_name = "varray";
                   4818: 
                   4819:   int_type_info  = type_info_init;
                   4820:   int_type_info.basic_type = bt_Int;
                   4821: 
                   4822:   void_type_info = type_info_init;
                   4823:   void_type_info.basic_type = bt_Void;
                   4824: 
                   4825:   while ((option = getopt (argc, argv, "d:i:I:o:v")) != EOF)
                   4826:     switch (option)
                   4827:       {
                   4828:       default:
                   4829:        had_errors++;
                   4830:        break;
                   4831: 
                   4832:       case 'd':
                   4833:        debug = strtol (optarg, &num_end, 0);
                   4834:        if ((unsigned)debug > 4 || num_end == optarg)
                   4835:          had_errors++;
                   4836: 
                   4837:        break;
                   4838: 
                   4839:       case 'I':
                   4840:        if (rename_output || obj_in_name != (char *)0)
                   4841:          had_errors++;
                   4842:        else
                   4843:          rename_output = 1;
                   4844: 
                   4845:        /* fall through to 'i' case.  */
                   4846: 
                   4847:       case 'i':
                   4848:        if (obj_in_name == (char *)0)
                   4849:          {
                   4850:            obj_in_name = optarg;
                   4851:            iflag++;
                   4852:          }
                   4853:        else
                   4854:          had_errors++;
                   4855:        break;
                   4856: 
                   4857:       case 'o':
                   4858:        if (object_name == (char *)0)
                   4859:          object_name = optarg;
                   4860:        else
                   4861:          had_errors++;
                   4862:        break;
                   4863: 
                   4864:       case 'v':
                   4865:        version++;
                   4866:        break;
                   4867:       }
                   4868: 
                   4869:   if (obj_in_name == (char *)0 && optind <= argc - 2)
                   4870:     obj_in_name = argv[--argc];
                   4871: 
                   4872:   if (object_name == (char *)0 && optind <= argc - 2)
                   4873:     object_name = argv[--argc];
                   4874: 
                   4875:   /* If there is an output name, but no input name use
                   4876:      the same file for both, deleting the name between
                   4877:      opening it for input and opening it for output.  */
                   4878:   if (obj_in_name == (char *)0 && object_name != (char *)0)
                   4879:     {
                   4880:       obj_in_name = object_name;
                   4881:       delete_input = 1;
                   4882:     }
                   4883: 
                   4884:   if (object_name == (char *)0 || had_errors || optind != argc - 1)
                   4885:     {
                   4886:       fprintf (stderr, "Calling Sequence:\n");
                   4887:       fprintf (stderr, "\tmips-tfile [-d <num>] [-v] [-i <o-in-file>] -o <o-out-file> <s-file> (or)\n");
                   4888:       fprintf (stderr, "\tmips-tfile [-d <num>] [-v] [-I <o-in-file>] -o <o-out-file> <s-file> (or)\n");
                   4889:       fprintf (stderr, "\tmips-tfile [-d <num>] [-v] <s-file> <o-in-file> <o-out-file>\n");
                   4890:       fprintf (stderr, "\n");
                   4891:       fprintf (stderr, "Debug levels are:\n");
                   4892:       fprintf (stderr, "    1\tGeneral debug + trace functions/blocks.\n");
                   4893:       fprintf (stderr, "    2\tDebug level 1 + trace externals.\n");
                   4894:       fprintf (stderr, "    3\tDebug level 2 + trace all symbols.\n");
                   4895:       fprintf (stderr, "    4\tDebug level 3 + trace memory allocations.\n");
                   4896:       return 1;
                   4897:     }
                   4898: 
                   4899: 
                   4900:   if (version)
                   4901:     {
                   4902:       fprintf (stderr, "mips-tfile version %s", version_string);
                   4903: #ifdef TARGET_VERSION
                   4904:       TARGET_VERSION;
                   4905: #endif
                   4906:       fputc ('\n', stderr);
                   4907:     }
                   4908: 
                   4909:   if (obj_in_name == (char *)0)
                   4910:     obj_in_name = object_name;
                   4911: 
                   4912:   if (rename_output && rename (object_name, obj_in_name) != 0)
                   4913:     {
                   4914:       char *buffer = (char *) allocate_multiple_pages (4);
                   4915:       int len;
                   4916:       int len2;
                   4917:       int in_fd;
                   4918:       int out_fd;
                   4919: 
                   4920:       /* Rename failed, copy input file */
                   4921:       in_fd = open (object_name, O_RDONLY, 0666);
                   4922:       if (in_fd < 0)
                   4923:        pfatal_with_name (object_name);
                   4924: 
                   4925:       out_fd = open (obj_in_name, O_WRONLY | O_CREAT | O_TRUNC, 0666);
                   4926:       if (out_fd < 0)
                   4927:        pfatal_with_name (obj_in_name);
                   4928: 
                   4929:       while ((len = read (in_fd, buffer, 4*PAGE_SIZE)) > 0)
                   4930:        {
                   4931:          len2 = write (out_fd, buffer, len);
                   4932:          if (len2 < 0)
                   4933:            pfatal_with_name (object_name);
                   4934: 
                   4935:          if (len != len2)
                   4936:            fatal ("wrote %d bytes to %s, expected to write %d", len2, obj_in_name, len);
                   4937:        }
                   4938: 
                   4939:       free_multiple_pages ((page_t *)buffer, 4);
                   4940: 
                   4941:       if (len < 0)
                   4942:        pfatal_with_name (object_name);
                   4943: 
                   4944:       if (close (in_fd) < 0)
                   4945:        pfatal_with_name (object_name);
                   4946: 
                   4947:       if (close (out_fd) < 0)
                   4948:        pfatal_with_name (obj_in_name);
                   4949:     }
                   4950: 
                   4951:   /* Must open input before output, since the output may be the same file, and
                   4952:      we need to get the input handle before truncating it.  */
                   4953:   obj_in_stream = fopen (obj_in_name, "r");
                   4954:   if (obj_in_stream == (FILE *)0)
                   4955:     pfatal_with_name (obj_in_name);
                   4956: 
                   4957:   if (delete_input && unlink (obj_in_name) != 0)
                   4958:     pfatal_with_name (obj_in_name);
                   4959: 
                   4960:   object_stream = fopen (object_name, "w");
                   4961:   if (object_stream == (FILE *)0)
                   4962:     pfatal_with_name (object_name);
                   4963: 
                   4964:   if (strcmp (argv[optind], "-") != 0)
                   4965:     {
                   4966:       input_name = argv[optind];
                   4967:       if (freopen (argv[optind], "r", stdin) != stdin)
                   4968:        pfatal_with_name (argv[optind]);
                   4969:     }
                   4970: 
                   4971:   copy_object ();                      /* scan & copy object file */
                   4972:   parse_input ();                      /* scan all of input */
                   4973: 
                   4974:   update_headers ();                   /* write out tfile */
                   4975:   write_object ();
                   4976: 
                   4977:   if (debug)
                   4978:     {
                   4979:       fprintf (stderr, "\n\tAllocation summary:\n\n");
                   4980:       for (i = (int)alloc_type_none; i < (int)alloc_type_last; i++)
                   4981:        if (alloc_counts[i].total_alloc)
                   4982:          {
                   4983:            fprintf (stderr,
                   4984:                     "\t%s\t%5d allocation(s), %5d free(s), %2d page(s)\n",
                   4985:                     alloc_counts[i].alloc_name,
                   4986:                     alloc_counts[i].total_alloc,
                   4987:                     alloc_counts[i].total_free,
                   4988:                     alloc_counts[i].total_pages);
                   4989:          }
                   4990:     }
                   4991: 
                   4992:   return (had_errors) ? 1 : 0;
                   4993: }
                   4994: 
                   4995: 
                   4996: /* Catch a signal and exit without dumping core.  */
                   4997: 
                   4998: STATIC void
                   4999: catch_signal (signum)
                   5000:      int signum;
                   5001: {
                   5002:   (void) signal (signum, SIG_DFL);     /* just in case... */
                   5003:   fatal (sys_siglist[signum]);
                   5004: }
                   5005: 
                   5006: /* Print a fatal error message.  NAME is the text.
                   5007:    Also include a system error message based on `errno'.  */
                   5008: 
                   5009: void
                   5010: pfatal_with_name (msg)
                   5011:      char *msg;
                   5012: {
                   5013:   int save_errno = errno;              /* just in case.... */
                   5014:   if (line_number > 0)
                   5015:     fprintf (stderr, "%s, %s:%ld ", progname, input_name, line_number);
                   5016:   else
                   5017:     fprintf (stderr, "%s:", progname);
                   5018: 
                   5019:   errno = save_errno;
                   5020:   if (errno == 0)
                   5021:     fprintf (stderr, "[errno = 0] %s\n", msg);
                   5022:   else
                   5023:     perror (msg);
                   5024: 
                   5025:   exit (1);
                   5026: }
                   5027: 
                   5028: 
                   5029: /* Procedure to abort with an out of bounds error message.  It has
                   5030:    type int, so it can be used with an ?: expression within the
                   5031:    ORIG_xxx macros, but the function never returns.  */
                   5032: 
                   5033: static int
                   5034: out_of_bounds (indx, max, str, prog_line)
                   5035:      symint_t indx;            /* index that is out of bounds */
                   5036:      symint_t max;             /* maximum index */
                   5037:      const char *str;          /* string to print out */
                   5038:      int prog_line;            /* line number within mips-tfile.c */
                   5039: {
                   5040:   if (indx < max)              /* just in case */
                   5041:     return 0;
                   5042: 
                   5043:   fprintf (stderr, "%s, %s:%ld index %u is out of bounds for %s, max is %u, mips-tfile.c line# %d\n",
                   5044:           progname, input_name, line_number, indx, str, max, prog_line);
                   5045: 
                   5046:   exit (1);
                   5047:   return 0;                    /* turn off warning messages */
                   5048: }
                   5049: 
                   5050: 
                   5051: /* Allocate a cluster of pages.  USE_MALLOC says that malloc does not
                   5052:    like sbrk's behind it's back (or sbrk isn't available).  If we use
                   5053:    sbrk, we assume it gives us zeroed pages.  */
                   5054: 
                   5055: #ifndef MALLOC_CHECK
                   5056: #ifdef USE_MALLOC
                   5057: 
                   5058: STATIC page_t *
                   5059: allocate_cluster (npages)
                   5060:      Size_t npages;
                   5061: {
                   5062:   register page_t *value = (page_t *) calloc (npages, PAGE_USIZE);
                   5063: 
                   5064:   if (value == 0)
                   5065:     fatal ("Virtual memory exhausted.");
                   5066: 
                   5067:   if (debug > 3)
                   5068:     fprintf (stderr, "\talloc\tnpages = %d, value = 0x%.8x\n", npages, value);
                   5069: 
                   5070:   return value;
                   5071: }
                   5072: 
                   5073: #else /* USE_MALLOC */
                   5074: 
                   5075: STATIC page_t *
                   5076: allocate_cluster (npages)
                   5077:      Size_t npages;
                   5078: {
                   5079:   register page_t *ptr = (page_t *) sbrk (0);  /* current sbreak */
                   5080:   unsigned long offset = ((unsigned long) ptr) & (PAGE_SIZE - 1);
                   5081: 
                   5082:   if (offset != 0)                     /* align to a page boundary */
                   5083:     {
                   5084:       if (sbrk (PAGE_USIZE - offset) == (char *)-1)
                   5085:        pfatal_with_name ("allocate_cluster");
                   5086: 
                   5087:       ptr = (page_t *) (((char *)ptr) + PAGE_SIZE - offset);
                   5088:     }
                   5089: 
                   5090:   if (sbrk (npages * PAGE_USIZE) == (char *)-1)
                   5091:     pfatal_with_name ("allocate_cluster");
                   5092: 
                   5093:   if (debug > 3)
                   5094:     fprintf (stderr, "\talloc\tnpages = %d, value = 0x%.8x\n", npages, ptr);
                   5095: 
                   5096:   return ptr;
                   5097: }
                   5098: 
                   5099: #endif /* USE_MALLOC */
                   5100: 
                   5101: 
                   5102: static page_t  *cluster_ptr    = NULL;
                   5103: static unsigned         pages_left     = 0;
                   5104: 
                   5105: #endif /* MALLOC_CHECK */
                   5106: 
                   5107: 
                   5108: /* Allocate some pages (which is initialized to 0).  */
                   5109: 
                   5110: STATIC page_t *
                   5111: allocate_multiple_pages (npages)
                   5112:      Size_t npages;
                   5113: {
                   5114: #ifndef MALLOC_CHECK
                   5115:   if (pages_left == 0 && npages < MAX_CLUSTER_PAGES)
                   5116:     {
                   5117:       pages_left = MAX_CLUSTER_PAGES;
                   5118:       cluster_ptr = allocate_cluster (MAX_CLUSTER_PAGES);
                   5119:     }
                   5120: 
                   5121:   if (npages <= pages_left)
                   5122:     {
                   5123:       page_t *ptr = cluster_ptr;
                   5124:       cluster_ptr += npages;
                   5125:       pages_left -= npages;
                   5126:       return ptr;
                   5127:     }
                   5128: 
                   5129:   return allocate_cluster (npages);
                   5130: 
                   5131: #else  /* MALLOC_CHECK */
                   5132:   return (page_t *) xcalloc (npages, PAGE_SIZE);
                   5133: 
                   5134: #endif /* MALLOC_CHECK */
                   5135: }
                   5136: 
                   5137: 
                   5138: /* Release some pages.  */
                   5139: 
                   5140: STATIC void
                   5141: free_multiple_pages (page_ptr, npages)
                   5142:      page_t *page_ptr;
                   5143:      Size_t npages;
                   5144: {
                   5145: #ifndef MALLOC_CHECK
                   5146:   if (pages_left == 0)
                   5147:     {
                   5148:       cluster_ptr = page_ptr;
                   5149:       pages_left = npages;
                   5150:     }
                   5151: 
                   5152:   else if ((page_ptr + npages) == cluster_ptr)
                   5153:     {
                   5154:       cluster_ptr -= npages;
                   5155:       pages_left += npages;
                   5156:     }
                   5157: 
                   5158:   /* otherwise the page is not freed.  If more than call is
                   5159:      done, we probably should worry about it, but at present,
                   5160:      the free pages is done right after an allocate.  */
                   5161: 
                   5162: #else  /* MALLOC_CHECK */
                   5163:   free ((char *) page_ptr);
                   5164: 
                   5165: #endif /* MALLOC_CHECK */
                   5166: }
                   5167: 
                   5168: 
                   5169: /* Allocate one page (which is initialized to 0).  */
                   5170: 
                   5171: STATIC page_t *
                   5172: allocate_page __proto((void))
                   5173: {
                   5174: #ifndef MALLOC_CHECK
                   5175:   if (pages_left == 0)
                   5176:     {
                   5177:       pages_left = MAX_CLUSTER_PAGES;
                   5178:       cluster_ptr = allocate_cluster (MAX_CLUSTER_PAGES);
                   5179:     }
                   5180: 
                   5181:   pages_left--;
                   5182:   return cluster_ptr++;
                   5183: 
                   5184: #else  /* MALLOC_CHECK */
                   5185:   return (page_t *) xcalloc (1, PAGE_SIZE);
                   5186: 
                   5187: #endif /* MALLOC_CHECK */
                   5188: }
                   5189: 
                   5190: 
                   5191: /* Allocate scoping information.  */
                   5192: 
                   5193: STATIC scope_t *
                   5194: allocate_scope __proto((void))
                   5195: {
                   5196:   register scope_t *ptr;
                   5197:   static scope_t initial_scope;
                   5198: 
                   5199: #ifndef MALLOC_CHECK
                   5200:   ptr = alloc_counts[ (int)alloc_type_scope ].free_list.f_scope;
                   5201:   if (ptr != (scope_t *)0)
                   5202:     alloc_counts[ (int)alloc_type_scope ].free_list.f_scope = ptr->free;
                   5203: 
                   5204:   else
                   5205:     {
                   5206:       register int unallocated = alloc_counts[ (int)alloc_type_scope ].unallocated;
                   5207:       register page_t *cur_page        = alloc_counts[ (int)alloc_type_scope ].cur_page;
                   5208: 
                   5209:       if (unallocated == 0)
                   5210:        {
                   5211:          unallocated = PAGE_SIZE / sizeof (scope_t);
                   5212:          alloc_counts[ (int)alloc_type_scope ].cur_page = cur_page = allocate_page ();
                   5213:          alloc_counts[ (int)alloc_type_scope ].total_pages++;
                   5214:        }
                   5215: 
                   5216:       ptr = &cur_page->scope[ --unallocated ];
                   5217:       alloc_counts[ (int)alloc_type_scope ].unallocated = unallocated;
                   5218:     }
                   5219: 
                   5220: #else
                   5221:   ptr = (scope_t *) xmalloc (sizeof (scope_t));
                   5222: 
                   5223: #endif
                   5224: 
                   5225:   alloc_counts[ (int)alloc_type_scope ].total_alloc++;
                   5226:   *ptr = initial_scope;
                   5227:   return ptr;
                   5228: }
                   5229: 
                   5230: /* Free scoping information.  */
                   5231: 
                   5232: STATIC void
                   5233: free_scope (ptr)
                   5234:      scope_t *ptr;
                   5235: {
                   5236:   alloc_counts[ (int)alloc_type_scope ].total_free++;
                   5237: 
                   5238: #ifndef MALLOC_CHECK
                   5239:   ptr->free = alloc_counts[ (int)alloc_type_scope ].free_list.f_scope;
                   5240:   alloc_counts[ (int)alloc_type_scope ].free_list.f_scope = ptr;
                   5241: 
                   5242: #else
                   5243:   xfree ((PTR_T) ptr);
                   5244: #endif
                   5245: 
                   5246: }
                   5247: 
                   5248: 
                   5249: /* Allocate links for pages in a virtual array.  */
                   5250: 
                   5251: STATIC vlinks_t *
                   5252: allocate_vlinks __proto((void))
                   5253: {
                   5254:   register vlinks_t *ptr;
                   5255:   static vlinks_t initial_vlinks;
                   5256: 
                   5257: #ifndef MALLOC_CHECK
                   5258:   register int unallocated     = alloc_counts[ (int)alloc_type_vlinks ].unallocated;
                   5259:   register page_t *cur_page    = alloc_counts[ (int)alloc_type_vlinks ].cur_page;
                   5260: 
                   5261:   if (unallocated == 0)
                   5262:     {
                   5263:       unallocated = PAGE_SIZE / sizeof (vlinks_t);
                   5264:       alloc_counts[ (int)alloc_type_vlinks ].cur_page = cur_page = allocate_page ();
                   5265:       alloc_counts[ (int)alloc_type_vlinks ].total_pages++;
                   5266:     }
                   5267: 
                   5268:   ptr = &cur_page->vlinks[ --unallocated ];
                   5269:   alloc_counts[ (int)alloc_type_vlinks ].unallocated = unallocated;
                   5270: 
                   5271: #else
                   5272:   ptr = (vlinks_t *) xmalloc (sizeof (vlinks_t));
                   5273: 
                   5274: #endif
                   5275: 
                   5276:   alloc_counts[ (int)alloc_type_vlinks ].total_alloc++;
                   5277:   *ptr = initial_vlinks;
                   5278:   return ptr;
                   5279: }
                   5280: 
                   5281: 
                   5282: /* Allocate string hash buckets.  */
                   5283: 
                   5284: STATIC shash_t *
                   5285: allocate_shash __proto((void))
                   5286: {
                   5287:   register shash_t *ptr;
                   5288:   static shash_t initial_shash;
                   5289: 
                   5290: #ifndef MALLOC_CHECK
                   5291:   register int unallocated     = alloc_counts[ (int)alloc_type_shash ].unallocated;
                   5292:   register page_t *cur_page    = alloc_counts[ (int)alloc_type_shash ].cur_page;
                   5293: 
                   5294:   if (unallocated == 0)
                   5295:     {
                   5296:       unallocated = PAGE_SIZE / sizeof (shash_t);
                   5297:       alloc_counts[ (int)alloc_type_shash ].cur_page = cur_page = allocate_page ();
                   5298:       alloc_counts[ (int)alloc_type_shash ].total_pages++;
                   5299:     }
                   5300: 
                   5301:   ptr = &cur_page->shash[ --unallocated ];
                   5302:   alloc_counts[ (int)alloc_type_shash ].unallocated = unallocated;
                   5303: 
                   5304: #else
                   5305:   ptr = (shash_t *) xmalloc (sizeof (shash_t));
                   5306: 
                   5307: #endif
                   5308: 
                   5309:   alloc_counts[ (int)alloc_type_shash ].total_alloc++;
                   5310:   *ptr = initial_shash;
                   5311:   return ptr;
                   5312: }
                   5313: 
                   5314: 
                   5315: /* Allocate type hash buckets.  */
                   5316: 
                   5317: STATIC thash_t *
                   5318: allocate_thash __proto((void))
                   5319: {
                   5320:   register thash_t *ptr;
                   5321:   static thash_t initial_thash;
                   5322: 
                   5323: #ifndef MALLOC_CHECK
                   5324:   register int unallocated     = alloc_counts[ (int)alloc_type_thash ].unallocated;
                   5325:   register page_t *cur_page    = alloc_counts[ (int)alloc_type_thash ].cur_page;
                   5326: 
                   5327:   if (unallocated == 0)
                   5328:     {
                   5329:       unallocated = PAGE_SIZE / sizeof (thash_t);
                   5330:       alloc_counts[ (int)alloc_type_thash ].cur_page = cur_page = allocate_page ();
                   5331:       alloc_counts[ (int)alloc_type_thash ].total_pages++;
                   5332:     }
                   5333: 
                   5334:   ptr = &cur_page->thash[ --unallocated ];
                   5335:   alloc_counts[ (int)alloc_type_thash ].unallocated = unallocated;
                   5336: 
                   5337: #else
                   5338:   ptr = (thash_t *) xmalloc (sizeof (thash_t));
                   5339: 
                   5340: #endif
                   5341: 
                   5342:   alloc_counts[ (int)alloc_type_thash ].total_alloc++;
                   5343:   *ptr = initial_thash;
                   5344:   return ptr;
                   5345: }
                   5346: 
                   5347: 
                   5348: /* Allocate structure, union, or enum tag information.  */
                   5349: 
                   5350: STATIC tag_t *
                   5351: allocate_tag __proto((void))
                   5352: {
                   5353:   register tag_t *ptr;
                   5354:   static tag_t initial_tag;
                   5355: 
                   5356: #ifndef MALLOC_CHECK
                   5357:   ptr = alloc_counts[ (int)alloc_type_tag ].free_list.f_tag;
                   5358:   if (ptr != (tag_t *)0)
                   5359:     alloc_counts[ (int)alloc_type_tag ].free_list.f_tag = ptr->free;
                   5360: 
                   5361:   else
                   5362:     {
                   5363:       register int unallocated = alloc_counts[ (int)alloc_type_tag ].unallocated;
                   5364:       register page_t *cur_page        = alloc_counts[ (int)alloc_type_tag ].cur_page;
                   5365: 
                   5366:       if (unallocated == 0)
                   5367:        {
                   5368:          unallocated = PAGE_SIZE / sizeof (tag_t);
                   5369:          alloc_counts[ (int)alloc_type_tag ].cur_page = cur_page = allocate_page ();
                   5370:          alloc_counts[ (int)alloc_type_tag ].total_pages++;
                   5371:        }
                   5372: 
                   5373:       ptr = &cur_page->tag[ --unallocated ];
                   5374:       alloc_counts[ (int)alloc_type_tag ].unallocated = unallocated;
                   5375:     }
                   5376: 
                   5377: #else
                   5378:   ptr = (tag_t *) xmalloc (sizeof (tag_t));
                   5379: 
                   5380: #endif
                   5381: 
                   5382:   alloc_counts[ (int)alloc_type_tag ].total_alloc++;
                   5383:   *ptr = initial_tag;
                   5384:   return ptr;
                   5385: }
                   5386: 
                   5387: /* Free scoping information.  */
                   5388: 
                   5389: STATIC void
                   5390: free_tag (ptr)
                   5391:      tag_t *ptr;
                   5392: {
                   5393:   alloc_counts[ (int)alloc_type_tag ].total_free++;
                   5394: 
                   5395: #ifndef MALLOC_CHECK
                   5396:   ptr->free = alloc_counts[ (int)alloc_type_tag ].free_list.f_tag;
                   5397:   alloc_counts[ (int)alloc_type_tag ].free_list.f_tag = ptr;
                   5398: 
                   5399: #else
                   5400:   xfree ((PTR_T) ptr);
                   5401: #endif
                   5402: 
                   5403: }
                   5404: 
                   5405: 
                   5406: /* Allocate forward reference to a yet unknown tag.  */
                   5407: 
                   5408: STATIC forward_t *
                   5409: allocate_forward __proto((void))
                   5410: {
                   5411:   register forward_t *ptr;
                   5412:   static forward_t initial_forward;
                   5413: 
                   5414: #ifndef MALLOC_CHECK
                   5415:   ptr = alloc_counts[ (int)alloc_type_forward ].free_list.f_forward;
                   5416:   if (ptr != (forward_t *)0)
                   5417:     alloc_counts[ (int)alloc_type_forward ].free_list.f_forward = ptr->free;
                   5418: 
                   5419:   else
                   5420:     {
                   5421:       register int unallocated = alloc_counts[ (int)alloc_type_forward ].unallocated;
                   5422:       register page_t *cur_page        = alloc_counts[ (int)alloc_type_forward ].cur_page;
                   5423: 
                   5424:       if (unallocated == 0)
                   5425:        {
                   5426:          unallocated = PAGE_SIZE / sizeof (forward_t);
                   5427:          alloc_counts[ (int)alloc_type_forward ].cur_page = cur_page = allocate_page ();
                   5428:          alloc_counts[ (int)alloc_type_forward ].total_pages++;
                   5429:        }
                   5430: 
                   5431:       ptr = &cur_page->forward[ --unallocated ];
                   5432:       alloc_counts[ (int)alloc_type_forward ].unallocated = unallocated;
                   5433:     }
                   5434: 
                   5435: #else
                   5436:   ptr = (forward_t *) xmalloc (sizeof (forward_t));
                   5437: 
                   5438: #endif
                   5439: 
                   5440:   alloc_counts[ (int)alloc_type_forward ].total_alloc++;
                   5441:   *ptr = initial_forward;
                   5442:   return ptr;
                   5443: }
                   5444: 
                   5445: /* Free scoping information.  */
                   5446: 
                   5447: STATIC void
                   5448: free_forward (ptr)
                   5449:      forward_t *ptr;
                   5450: {
                   5451:   alloc_counts[ (int)alloc_type_forward ].total_free++;
                   5452: 
                   5453: #ifndef MALLOC_CHECK
                   5454:   ptr->free = alloc_counts[ (int)alloc_type_forward ].free_list.f_forward;
                   5455:   alloc_counts[ (int)alloc_type_forward ].free_list.f_forward = ptr;
                   5456: 
                   5457: #else
                   5458:   xfree ((PTR_T) ptr);
                   5459: #endif
                   5460: 
                   5461: }
                   5462: 
                   5463: 
                   5464: /* Allocate head of type hash list.  */
                   5465: 
                   5466: STATIC thead_t *
                   5467: allocate_thead __proto((void))
                   5468: {
                   5469:   register thead_t *ptr;
                   5470:   static thead_t initial_thead;
                   5471: 
                   5472: #ifndef MALLOC_CHECK
                   5473:   ptr = alloc_counts[ (int)alloc_type_thead ].free_list.f_thead;
                   5474:   if (ptr != (thead_t *)0)
                   5475:     alloc_counts[ (int)alloc_type_thead ].free_list.f_thead = ptr->free;
                   5476: 
                   5477:   else
                   5478:     {
                   5479:       register int unallocated = alloc_counts[ (int)alloc_type_thead ].unallocated;
                   5480:       register page_t *cur_page        = alloc_counts[ (int)alloc_type_thead ].cur_page;
                   5481: 
                   5482:       if (unallocated == 0)
                   5483:        {
                   5484:          unallocated = PAGE_SIZE / sizeof (thead_t);
                   5485:          alloc_counts[ (int)alloc_type_thead ].cur_page = cur_page = allocate_page ();
                   5486:          alloc_counts[ (int)alloc_type_thead ].total_pages++;
                   5487:        }
                   5488: 
                   5489:       ptr = &cur_page->thead[ --unallocated ];
                   5490:       alloc_counts[ (int)alloc_type_thead ].unallocated = unallocated;
                   5491:     }
                   5492: 
                   5493: #else
                   5494:   ptr = (thead_t *) xmalloc (sizeof (thead_t));
                   5495: 
                   5496: #endif
                   5497: 
                   5498:   alloc_counts[ (int)alloc_type_thead ].total_alloc++;
                   5499:   *ptr = initial_thead;
                   5500:   return ptr;
                   5501: }
                   5502: 
                   5503: /* Free scoping information.  */
                   5504: 
                   5505: STATIC void
                   5506: free_thead (ptr)
                   5507:      thead_t *ptr;
                   5508: {
                   5509:   alloc_counts[ (int)alloc_type_thead ].total_free++;
                   5510: 
                   5511: #ifndef MALLOC_CHECK
                   5512:   ptr->free = (thead_t *) alloc_counts[ (int)alloc_type_thead ].free_list.f_thead;
                   5513:   alloc_counts[ (int)alloc_type_thead ].free_list.f_thead = ptr;
                   5514: 
                   5515: #else
                   5516:   xfree ((PTR_T) ptr);
                   5517: #endif
                   5518: 
                   5519: }
                   5520: 
                   5521: #endif /* MIPS_DEBUGGING_INFO */
                   5522: 
                   5523: 
                   5524: /* Output an error message and exit */
                   5525: 
                   5526: /*VARARGS*/
                   5527: void
                   5528: fatal (va_alist)
                   5529:      va_dcl
                   5530: {
                   5531:   va_list ap;
                   5532:   char *format;
                   5533: 
                   5534:   if (line_number > 0)
                   5535:     fprintf (stderr, "%s, %s:%ld ", progname, input_name, line_number);
                   5536:   else
                   5537:     fprintf (stderr, "%s:", progname);
                   5538: 
                   5539:   va_start(ap);
                   5540:   format = va_arg (ap, char *);
                   5541:   vfprintf (stderr, format, ap);
                   5542:   va_end (ap);
                   5543:   fprintf (stderr, "\n");
                   5544:   if (line_number > 0)
                   5545:     fprintf (stderr, "line:\t%s\n", cur_line_start);
                   5546: 
                   5547:   saber_stop ();
                   5548:   exit (1);
                   5549: }
                   5550: 
                   5551: /*VARARGS*/
                   5552: void
                   5553: error (va_alist) 
                   5554:      va_dcl
                   5555: {
                   5556:   va_list ap;
                   5557:   char *format;
                   5558: 
                   5559:   if (line_number > 0)
                   5560:     fprintf (stderr, "%s, %s:%ld ", progname, input_name, line_number);
                   5561:   else
                   5562:     fprintf (stderr, "%s:", progname);
                   5563: 
                   5564:   va_start(ap);
                   5565:   format = va_arg (ap, char *);
                   5566:   vfprintf (stderr, format, ap);
                   5567:   fprintf (stderr, "\n");
                   5568:   if (line_number > 0)
                   5569:     fprintf (stderr, "line:\t%s\n", cur_line_start);
                   5570: 
                   5571:   had_errors++;
                   5572:   va_end (ap);
                   5573: 
                   5574:   saber_stop ();
                   5575: }
                   5576: 
                   5577: /* More 'friendly' abort that prints the line and file.
                   5578:    config.h can #define abort fancy_abort if you like that sort of thing.  */
                   5579: 
                   5580: void
                   5581: fancy_abort ()
                   5582: {
                   5583:   fatal ("Internal abort.");
                   5584: }
                   5585: 
                   5586: 
                   5587: /* When `malloc.c' is compiled with `rcheck' defined,
                   5588:    it calls this function to report clobberage.  */
                   5589: 
                   5590: void
                   5591: botch (s)
                   5592:      const char *s;
                   5593: {
                   5594:   fatal (s);
                   5595: }
                   5596: 
                   5597: /* Same as `malloc' but report error if no memory available.  */
                   5598: 
                   5599: PTR_T
                   5600: xmalloc (size)
                   5601:      Size_t size;
                   5602: {
                   5603:   register PTR_T value = malloc (size);
                   5604:   if (value == 0)
                   5605:     fatal ("Virtual memory exhausted.");
                   5606: 
                   5607:   if (debug > 3)
                   5608:     fprintf (stderr, "\tmalloc\tptr = 0x%.8x, size = %10u\n", value, size);
                   5609: 
                   5610:   return value;
                   5611: }
                   5612: 
                   5613: /* Same as `calloc' but report error if no memory available.  */
                   5614: 
                   5615: PTR_T
                   5616: xcalloc (size1, size2)
                   5617:      Size_t size1, size2;
                   5618: {
                   5619:   register PTR_T value = calloc (size1, size2);
                   5620:   if (value == 0)
                   5621:     fatal ("Virtual memory exhausted.");
                   5622: 
                   5623:   if (debug > 3)
                   5624:     fprintf (stderr, "\tcalloc\tptr = 0x%.8x, size1 = %10u, size2 = %10u [%u]\n",
                   5625:             value, size1, size2, size1+size2);
                   5626: 
                   5627:   return value;
                   5628: }
                   5629: 
                   5630: /* Same as `realloc' but report error if no memory available.  */
                   5631: 
                   5632: PTR_T
                   5633: xrealloc (ptr, size)
                   5634:      PTR_T ptr;
                   5635:      Size_t size;
                   5636: {
                   5637:   register PTR_T result = realloc (ptr, size);
                   5638:   if (!result)
                   5639:     fatal ("Virtual memory exhausted.");
                   5640: 
                   5641:   if (debug > 3)
                   5642:     fprintf (stderr, "\trealloc\tptr = 0x%.8x, size = %10u, orig = 0x%.8x\n",
                   5643:             result, size, ptr);
                   5644: 
                   5645:   return result;
                   5646: }
                   5647: 
                   5648: void
                   5649: xfree (ptr)
                   5650:      PTR_T ptr;
                   5651: {
                   5652:   if (debug > 3)
                   5653:     fprintf (stderr, "\tfree\tptr = 0x%.8x\n", ptr);
                   5654: 
                   5655:   free (ptr);
                   5656: }
                   5657: 
                   5658: 
                   5659: /* Define our own index/rindex, since the local and global symbol
                   5660:    structures as defined by MIPS has an 'index' field.  */
                   5661: 
                   5662: STATIC char *
                   5663: local_index (str, sentinel)
                   5664:      const char *str;
                   5665:      int sentinel;
                   5666: {
                   5667:   int ch;
                   5668: 
                   5669:   for ( ; (ch = *str) != sentinel; str++)
                   5670:     {
                   5671:       if (ch == '\0')
                   5672:        return (char *)0;
                   5673:     }
                   5674: 
                   5675:   return (char *)str;
                   5676: }
                   5677: 
                   5678: STATIC char *
                   5679: local_rindex (str, sentinel)
                   5680:      const char *str;
                   5681:      int sentinel;
                   5682: {
                   5683:   int ch;
                   5684:   const char *ret = (const char *)0;
                   5685: 
                   5686:   for ( ; (ch = *str) != '\0'; str++)
                   5687:     {
                   5688:       if (ch == sentinel)
                   5689:        ret = str;
                   5690:     }
                   5691: 
                   5692:   return (char *)ret;
                   5693: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.