diff options
-rw-r--r-- | gas/ChangeLog | 3 | ||||
-rw-r--r-- | gas/config/tc-dvp.c | 101 |
2 files changed, 72 insertions, 32 deletions
diff --git a/gas/ChangeLog b/gas/ChangeLog index 7b7a4de..208a282 100644 --- a/gas/ChangeLog +++ b/gas/ChangeLog @@ -3,6 +3,9 @@ Tue Feb 24 11:01:25 1998 Doug Evans <devans@canuck.cygnus.com> * config/tc-dvp.h (tc_frob_label): Define. * config/tc-dvp.c (dvp_frob_label): New function. + (create_colon_label): New argument `sto'. All callers updated. + (record_mach): New argument `force_next_p'. All callers updated. + (cpu_sto): New function. Mon Feb 23 15:55:12 1998 Doug Evans <devans@canuck.cygnus.com> diff --git a/gas/config/tc-dvp.c b/gas/config/tc-dvp.c index bc33aaf..d03c6c5 100644 --- a/gas/config/tc-dvp.c +++ b/gas/config/tc-dvp.c @@ -45,7 +45,7 @@ static long parse_float PARAMS ((char **, const char **)); static symbolS * create_label PARAMS ((const char *, const char *)); -static symbolS * create_colon_label PARAMS ((const char *, const char *)); +static symbolS * create_colon_label PARAMS ((int, const char *, const char *)); static char * unique_name PARAMS ((const char *)); static long eval_expr PARAMS ((int, int, const char *, ...)); static long parse_dma_addr_autocount (); @@ -101,13 +101,15 @@ static void push_asm_state PARAMS ((asm_state)); static void pop_asm_state PARAMS ((int)); static void set_asm_state PARAMS ((asm_state)); -/* Current mach (machine variant) type state. +/* Current cpu (machine variant) type state. We copy the mips16 way of recording what the current machine type is in the code. A label is created whenever necessary and has an "other" value the denotes the machine type. */ -static dvp_cpu cur_mach; +static dvp_cpu cur_cpu; /* Record the current mach type. */ -static void record_mach PARAMS ((dvp_cpu)); +static void record_mach PARAMS ((dvp_cpu, int)); +/* Given a dvp_cpu value, return the STO_DVP value to use. */ +static int cpu_sto PARAMS ((dvp_cpu, const char **)); /* Nonzero if inside .DmaData. */ static int dma_data_state = 0; @@ -224,7 +226,7 @@ md_begin () /* Set the current mach to an illegal value to force a label for the first insn. */ - cur_mach = -1; + cur_cpu = DVP_UNKNOWN; /* Initialize the parsing state. */ cur_state_index = 0; @@ -362,7 +364,7 @@ assemble_dma (str) if (!output_dma) return; - record_mach (DVP_DMA); + record_mach (DVP_DMA, 0); len = 4; f = frag_more (len * 4); @@ -434,9 +436,15 @@ assemble_vif (str) if (output_vif) { /* Record the mach before doing the alignment so that we properly - disassemble any inserted vifnop's. */ + disassemble any inserted vifnop's. For variable length insns + force the recording of the mach type for the next insn. A label may + be embedded in it to compute the length and this will cause the + disassembler to wrongly disassemble the next insn. */ - record_mach (DVP_VIF); + if (opcode->flags & VIF_OPCODE_LENVAR) + record_mach (DVP_VIF, 1); + else + record_mach (DVP_VIF, 0); if (opcode->flags & VIF_OPCODE_MPG) { @@ -558,7 +566,7 @@ assemble_gif (str) frag_align (4, 0, 0); record_alignment (now_seg, 4); - record_mach (DVP_GIF); + record_mach (DVP_GIF, 1); gif_insn_frag = f = frag_more (16); for (i = 0; i < 4; ++i) @@ -566,7 +574,7 @@ assemble_gif (str) /* Insert a label so we can compute the number of quadwords when the .endgif is seen. */ - gif_data_name = S_GET_NAME (create_colon_label ("", unique_name (NULL))); + gif_data_name = S_GET_NAME (create_colon_label (0, "", unique_name (NULL))); /* Record the type of the gif tag so we know how to compute nloop in s_endgif. */ @@ -602,7 +610,7 @@ assemble_vu (str) frag_align (3, 0, 0); record_alignment (now_seg, 3); - record_mach (DVP_VUUP); + record_mach (DVP_VUUP, 0); /* The lower instruction has the lower address. Handle this by grabbing 8 bytes now, and then filling each word @@ -968,31 +976,49 @@ assemble_one_insn (cpu, opcode, operand_table, pstr, insn_buf) return 0; } -/* Record the current mach type in the object file. */ +/* Given a dvp cpu type, return it's STO_DVP value. + The section name prefix to use is stored in *PNAME. */ + +static int +cpu_sto (cpu, pname) + dvp_cpu cpu; + const char **pname; +{ + switch (cpu) + { + case DVP_DMA : *pname = ".dma."; return STO_DVP_DMA; + case DVP_VIF : *pname = ".vif."; return STO_DVP_VIF; + case DVP_GIF : *pname = ".gif."; return STO_DVP_GIF; + case DVP_VUUP : *pname = ".vu."; return STO_DVP_VU; + } + abort (); +} + +/* Record the current mach type in the object file. + If FORCE_NEXT_P is non-zero, force a label to be emitted the next time + we're called. This is useful for variable length instructions that can + have labels embedded within them. */ static void -record_mach (mach) - dvp_cpu mach; +record_mach (cpu, force_next_p) + dvp_cpu cpu; + int force_next_p; { symbolS *label; - char *name; - int other; + const char *name; + int sto; - if (mach == cur_mach) + if (cpu == cur_cpu) return; - switch (mach) - { - case DVP_DMA : name = ".dma."; other = STO_DVP_DMA; break; - case DVP_VIF : name = ".vif."; other = STO_DVP_VIF; break; - case DVP_GIF : name = ".gif."; other = STO_DVP_GIF; break; - case DVP_VUUP : name = ".vu."; other = STO_DVP_VU; break; - default : abort (); - } + sto = cpu_sto (cpu, &name); + + label = create_colon_label (sto, "", unique_name (name)); - label = create_colon_label ("", unique_name (name)); - S_SET_OTHER (label, other); - cur_mach = mach; + if (force_next_p) + cur_cpu = DVP_UNKNOWN; + else + cur_cpu = cpu; } /* Push/pop the current parsing state. */ @@ -1078,6 +1104,9 @@ void dvp_frob_label (sym) symbolS *sym; { + /* All labels in vu code must be specially marked for the disassembler. + The disassembler ignores all previous information at each new label + (that has a higher address than the last one). */ if (CUR_ASM_STATE == ASM_MPG || CUR_ASM_STATE == ASM_VU) S_SET_OTHER (sym, STO_DVP_VU); @@ -1487,10 +1516,14 @@ create_label (prefix, name) } /* Create a label named by concatenating PREFIX to NAME, - and define it as `.'. */ + and define it as `.'. + STO, if non-zero, is the st_other value to assign to this label. + If STO is zero `cur_cpu' is set to DVP_UNKNOWN to force record_mach to + emit a cpu label. Otherwise the disassembler gets confused. */ static symbolS * -create_colon_label (prefix, name) +create_colon_label (sto, prefix, name) + int sto; const char *prefix, *name; { int namelen = strlen (name); @@ -1502,6 +1535,10 @@ create_colon_label (prefix, name) strcpy (fullname, prefix); strcat (fullname, name); result = colon (fullname); + if (sto) + S_SET_OTHER (result, sto); + else + cur_cpu = DVP_UNKNOWN; free (fullname); return result; } @@ -1573,7 +1610,7 @@ inline_dma_data (autocount_p, insn_buf) if (autocount_p) { - dma_data_name = S_GET_NAME (create_colon_label ("", unique_name (NULL))); + dma_data_name = S_GET_NAME (create_colon_label (0, "", unique_name (NULL))); setup_dma_autocount (dma_data_name, insn_buf, 1); } else @@ -1920,7 +1957,7 @@ s_enddmadata (ignore) /* Fill the data out to a multiple of 16 bytes. */ /* FIXME: Are the fill contents right? */ frag_align (4, 0, 0); - create_colon_label (END_LABEL_PREFIX, dma_data_name); + create_colon_label (0, END_LABEL_PREFIX, dma_data_name); } } |