diff options
author | bernds <bernds@138bc75d-0d04-0410-961f-82ee72b054a4> | 2011-09-02 17:53:54 +0000 |
---|---|---|
committer | bernds <bernds@138bc75d-0d04-0410-961f-82ee72b054a4> | 2011-09-02 17:53:54 +0000 |
commit | 87ded6872697a5b3b114b23be7b41cac398df201 (patch) | |
tree | c86b8a712bb7f6e6f92e28398c0e34fa6d44b3c4 /gcc/config | |
parent | 01590c6f02af5de7844c7962b367fa7ae57fab30 (diff) | |
download | gcc-87ded6872697a5b3b114b23be7b41cac398df201.tar.gz |
* config/c6x/c6x.md (collapse-ndfa, no-comb-vect): New
automata_options.
(d1, l1, s1, m1, d2, l2, s2, m2): Changed to define_query_cpu_unit.
(l1w, s1w, l2w, s2w): Define in the main automaton.
(fps1, fpl1, adddps1, adddpl1, fps2, fpl2, adddps2, adddpl2): New
units.
* config/c6x/c6x.c (c6x_sched_insn_info): Add unit_mask member.
(c6x_unit_names): Add the new units.
(c6x_unit_codes): New static array.
(UNIT_QID_D1, UNIT_QID_L1, UNIT_QID_S1, UNIT_QID_M1, UNIT_QID_FPS1,
UNIT_QID_FPL1, UNIT_QID_ADDDPS1, UNIT_QID_ADDDPL1,
UNIT_QID_SIDE_OFFSET): New macros.
(RESERVATION_S2): Adjust value.
(c6x_option_override): Compute c6x_unit_codes.
(assign_reservations): Take the unit_mask of the last instruction
into account. Detect floating point reservations by looking for
the new units. Don't assign reservations if the field is already
nonzero.
(struct c6x_sched_context): Add member prev_cycle_state_ctx.
(init_sched_state): Initialize it.
(c6x_clear_sched_context): Free it.
(insn_set_clock): Clear reservation.
(prev_cycle_state): New static variable.
(c6x_init_sched_context): Save it.
(c6x_sched_init): Allocate space for it and clear it.
(c6x_sched_dfa_pre_cycle_insn): New static function.
(c6x_dfa_new_cycle): Save state at the start of a new cycle.
(c6x_variable_issue): Only record units in the unit_mask that
were not set at the start of the cycle.
(c6x_variable_issue): Compute and store the unit_mask from the
current state.
(reorg_split_calls): Ensure the new information remains correct.
(TARGET_SCHED_DFA_NEW_CYCLE, TARGET_SCHED_CLEAR_SCHED_CONTEXT,
TARGET_SCHED_DFA_PRE_CYCLE_INSN): Define.
* config/c6x/c6x.h (CPU_UNITS_QUERY): Define.
* config/c6x/c6x-sched.md.in (fp4_ls_N__CROSS_, adddp_ls_N__CROSS_):
Add special reservations.
* config/c6x/c6x-sched.md: Regenerate.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@178488 138bc75d-0d04-0410-961f-82ee72b054a4
Diffstat (limited to 'gcc/config')
-rw-r--r-- | gcc/config/c6x/c6x-sched.md | 16 | ||||
-rw-r--r-- | gcc/config/c6x/c6x-sched.md.in | 4 | ||||
-rw-r--r-- | gcc/config/c6x/c6x.c | 235 | ||||
-rw-r--r-- | gcc/config/c6x/c6x.h | 2 | ||||
-rw-r--r-- | gcc/config/c6x/c6x.md | 20 |
5 files changed, 236 insertions, 41 deletions
diff --git a/gcc/config/c6x/c6x-sched.md b/gcc/config/c6x/c6x-sched.md index 6cb4b66acd5..dc8e39cf452 100644 --- a/gcc/config/c6x/c6x-sched.md +++ b/gcc/config/c6x/c6x-sched.md @@ -183,14 +183,14 @@ (and (eq_attr "cross" "n") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "a")))) - "(s1,nothing*2,s1w)|(l1,nothing*2,l1w)") + "(fps1+s1,nothing*2,s1w)|(fpl1+l1,nothing*2,l1w)") (define_insn_reservation "adddp_ls1n" 7 (and (eq_attr "type" "adddp") (and (eq_attr "cross" "n") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "a")))) - "((s1)*2,nothing*3,s1w*2)|((l1)*2,nothing*3,l1w*2)") + "(adddps1+(s1)*2,nothing*3,s1w*2)|(adddpl1+(l1)*2,nothing*3,l1w*2)") (define_insn_reservation "single_dls1n" 1 (and (eq_attr "type" "single") @@ -416,14 +416,14 @@ (and (eq_attr "cross" "n") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "b")))) - "(s2,nothing*2,s2w)|(l2,nothing*2,l2w)") + "(fps2+s2,nothing*2,s2w)|(fpl2+l2,nothing*2,l2w)") (define_insn_reservation "adddp_ls2n" 7 (and (eq_attr "type" "adddp") (and (eq_attr "cross" "n") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "b")))) - "((s2)*2,nothing*3,s2w*2)|((l2)*2,nothing*3,l2w*2)") + "(adddps2+(s2)*2,nothing*3,s2w*2)|(adddpl2+(l2)*2,nothing*3,l2w*2)") (define_insn_reservation "single_dls2n" 1 (and (eq_attr "type" "single") @@ -649,14 +649,14 @@ (and (eq_attr "cross" "y") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "a")))) - "(s1+x1,nothing*2,s1w)|(l1+x1,nothing*2,l1w)") + "(fps1+s1+x1,nothing*2,s1w)|(fpl1+l1+x1,nothing*2,l1w)") (define_insn_reservation "adddp_ls1y" 7 (and (eq_attr "type" "adddp") (and (eq_attr "cross" "y") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "a")))) - "((s1+x1)*2,nothing*3,s1w*2)|((l1+x1)*2,nothing*3,l1w*2)") + "(adddps1+(s1+x1)*2,nothing*3,s1w*2)|(adddpl1+(l1+x1)*2,nothing*3,l1w*2)") (define_insn_reservation "single_dls1y" 1 (and (eq_attr "type" "single") @@ -882,14 +882,14 @@ (and (eq_attr "cross" "y") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "b")))) - "(s2+x2,nothing*2,s2w)|(l2+x2,nothing*2,l2w)") + "(fps2+s2+x2,nothing*2,s2w)|(fpl2+l2+x2,nothing*2,l2w)") (define_insn_reservation "adddp_ls2y" 7 (and (eq_attr "type" "adddp") (and (eq_attr "cross" "y") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "b")))) - "((s2+x2)*2,nothing*3,s2w*2)|((l2+x2)*2,nothing*3,l2w*2)") + "(adddps2+(s2+x2)*2,nothing*3,s2w*2)|(adddpl2+(l2+x2)*2,nothing*3,l2w*2)") (define_insn_reservation "single_dls2y" 1 (and (eq_attr "type" "single") diff --git a/gcc/config/c6x/c6x-sched.md.in b/gcc/config/c6x/c6x-sched.md.in index 271109b9cf5..0ba71433878 100644 --- a/gcc/config/c6x/c6x-sched.md.in +++ b/gcc/config/c6x/c6x-sched.md.in @@ -178,14 +178,14 @@ (and (eq_attr "cross" "_CROSS_") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "_RF_")))) - "(s_N__CUNIT_,nothing*2,s_N_w)|(l_N__CUNIT_,nothing*2,l_N_w)") + "(fps_N_+s_N__CUNIT_,nothing*2,s_N_w)|(fpl_N_+l_N__CUNIT_,nothing*2,l_N_w)") (define_insn_reservation "adddp_ls_N__CROSS_" 7 (and (eq_attr "type" "adddp") (and (eq_attr "cross" "_CROSS_") (and (eq_attr "units" "ls") (eq_attr "dest_regfile" "_RF_")))) - "((s_N__CUNIT_)*2,nothing*3,s_N_w*2)|((l_N__CUNIT_)*2,nothing*3,l_N_w*2)") + "(adddps_N_+(s_N__CUNIT_)*2,nothing*3,s_N_w*2)|(adddpl_N_+(l_N__CUNIT_)*2,nothing*3,l_N_w*2)") (define_insn_reservation "single_dls_N__CROSS_" 1 (and (eq_attr "type" "single") diff --git a/gcc/config/c6x/c6x.c b/gcc/config/c6x/c6x.c index e77ed70fd35..fd72babd2ae 100644 --- a/gcc/config/c6x/c6x.c +++ b/gcc/config/c6x/c6x.c @@ -111,6 +111,10 @@ typedef struct rtx new_cond; /* True for the first insn that was scheduled in an ebb. */ bool ebb_start; + /* The scheduler state after the insn, transformed into a mask of UNIT_QID + bits rather than storing the state. Meaningful only for the last + insn in a cycle. */ + unsigned int unit_mask; } c6x_sched_insn_info; DEF_VEC_O(c6x_sched_insn_info); @@ -124,13 +128,6 @@ static VEC(c6x_sched_insn_info, heap) *insn_info; static bool done_cfi_sections; -/* The DFA names of the units, in packet order. */ -static const char *const c6x_unit_names[] = -{ - "d1", "l1", "s1", "m1", - "d2", "l2", "s2", "m2", -}; - #define RESERVATION_FLAG_D 1 #define RESERVATION_FLAG_L 2 #define RESERVATION_FLAG_S 4 @@ -140,8 +137,29 @@ static const char *const c6x_unit_names[] = #define RESERVATION_FLAG_LS (RESERVATION_FLAG_L | RESERVATION_FLAG_S) #define RESERVATION_FLAG_DLS (RESERVATION_FLAG_D | RESERVATION_FLAG_LS) +/* The DFA names of the units. */ +static const char *const c6x_unit_names[] = +{ + "d1", "l1", "s1", "m1", "fps1", "fpl1", "adddps1", "adddpl1", + "d2", "l2", "s2", "m2", "fps2", "fpl2", "adddps2", "adddpl2" +}; + +/* The DFA unit number for each unit in c6x_unit_names[]. */ +static int c6x_unit_codes[ARRAY_SIZE (c6x_unit_names)]; + +/* Unit query IDs. */ +#define UNIT_QID_D1 0 +#define UNIT_QID_L1 1 +#define UNIT_QID_S1 2 +#define UNIT_QID_M1 3 +#define UNIT_QID_FPS1 4 +#define UNIT_QID_FPL1 5 +#define UNIT_QID_ADDDPS1 6 +#define UNIT_QID_ADDDPL1 7 +#define UNIT_QID_SIDE_OFFSET 8 + #define RESERVATION_S1 2 -#define RESERVATION_S2 6 +#define RESERVATION_S2 10 /* Register map for debugging. */ int const dbx_register_map[FIRST_PSEUDO_REGISTER] = @@ -169,6 +187,8 @@ c6x_init_machine_status (void) static void c6x_option_override (void) { + unsigned i; + if (global_options_set.x_c6x_arch_option) { c6x_arch = all_isas[c6x_arch_option].type; @@ -184,6 +204,9 @@ c6x_option_override (void) init_machine_status = c6x_init_machine_status; + for (i = 0; i < ARRAY_SIZE (c6x_unit_names); i++) + c6x_unit_codes[i] = get_cpu_unit_code (c6x_unit_names[i]); + if (flag_pic && !TARGET_DSBT) { error ("-fpic and -fPIC not supported without -mdsbt on this target"); @@ -2990,16 +3013,56 @@ assign_reservations (rtx head, rtx end) rtx insn; for (insn = head; insn != NEXT_INSN (end); insn = NEXT_INSN (insn)) { - rtx within; + unsigned int sched_mask, reserved; + rtx within, last; int pass; int rsrv[2]; int rsrv_count[2][4]; + int i; if (GET_MODE (insn) != TImode) continue; - rsrv[0] = rsrv[1] = 0; + reserved = 0; + last = NULL_RTX; + /* Find the last insn in the packet. It has a state recorded for it, + which we can use to determine the units we should be using. */ + for (within = insn; + (within != NEXT_INSN (end) + && (within == insn || GET_MODE (within) != TImode)); + within = NEXT_INSN (within)) + { + int icode; + if (!NONDEBUG_INSN_P (within)) + continue; + icode = recog_memoized (within); + if (icode < 0) + continue; + if (shadow_p (within)) + continue; + if (INSN_INFO_ENTRY (INSN_UID (within)).reservation != 0) + reserved |= 1 << INSN_INFO_ENTRY (INSN_UID (within)).reservation; + last = within; + } + if (last == NULL_RTX) + continue; + + sched_mask = INSN_INFO_ENTRY (INSN_UID (last)).unit_mask; + sched_mask &= ~reserved; + memset (rsrv_count, 0, sizeof rsrv_count); + rsrv[0] = rsrv[1] = ~0; + for (i = 0; i < 8; i++) + { + int side = i / 4; + int unit = i & 3; + unsigned unit_bit = 1 << (unit + side * UNIT_QID_SIDE_OFFSET); + /* Clear the bits which we expect to reserve in the following loop, + leaving the ones set which aren't present in the scheduler's + state and shouldn't be reserved. */ + if (sched_mask & unit_bit) + rsrv[i / 4] &= ~(1 << unit); + } /* Walk through the insns that occur in the same cycle. We use multiple passes to assign units, assigning for insns with the most specific @@ -3010,9 +3073,11 @@ assign_reservations (rtx head, rtx end) && (within == insn || GET_MODE (within) != TImode)); within = NEXT_INSN (within)) { + int uid = INSN_UID (within); int this_rsrv, side; int icode; enum attr_units units; + enum attr_type type; int j; if (!NONDEBUG_INSN_P (within)) @@ -3020,17 +3085,44 @@ assign_reservations (rtx head, rtx end) icode = recog_memoized (within); if (icode < 0) continue; + if (INSN_INFO_ENTRY (uid).reservation != 0) + continue; units = get_attr_units (within); + type = get_attr_type (within); this_rsrv = get_reservation_flags (units); if (this_rsrv == 0) continue; side = get_insn_side (within, units); + /* Certain floating point instructions are treated specially. If + an insn can choose between units it can reserve, and its + reservation spans more than one cycle, the reservation contains + special markers in the first cycle to help us reconstruct what + the automaton chose. */ + if ((type == TYPE_ADDDP || type == TYPE_FP4) + && units == UNITS_LS) + { + int test1_code = ((type == TYPE_FP4 ? UNIT_QID_FPL1 : UNIT_QID_ADDDPL1) + + side * UNIT_QID_SIDE_OFFSET); + int test2_code = ((type == TYPE_FP4 ? UNIT_QID_FPS1 : UNIT_QID_ADDDPS1) + + side * UNIT_QID_SIDE_OFFSET); + if ((sched_mask & (1 << test1_code)) != 0) + { + this_rsrv = RESERVATION_FLAG_L; + sched_mask &= ~(1 << test1_code); + } + else if ((sched_mask & (1 << test2_code)) != 0) + { + this_rsrv = RESERVATION_FLAG_S; + sched_mask &= ~(1 << test2_code); + } + } + if ((this_rsrv & (this_rsrv - 1)) == 0) { - int t = exact_log2 (this_rsrv) + side * 4; + int t = exact_log2 (this_rsrv) + side * UNIT_QID_SIDE_OFFSET; rsrv[side] |= this_rsrv; - INSN_INFO_ENTRY (INSN_UID (within)).reservation = t; + INSN_INFO_ENTRY (uid).reservation = t; continue; } @@ -3059,8 +3151,8 @@ assign_reservations (rtx head, rtx end) if ((this_rsrv & (1 << j)) && j != best) rsrv_count[side][j]--; - INSN_INFO_ENTRY (INSN_UID (within)).reservation - = best + side * 4; + INSN_INFO_ENTRY (uid).reservation + = best + side * UNIT_QID_SIDE_OFFSET; } } } @@ -3098,6 +3190,12 @@ typedef struct c6x_sched_context /* The following variable value is the last issued insn. */ rtx last_scheduled_insn; + /* The following variable value is DFA state before issuing the + first insn in the current clock cycle. We do not use this member + of the structure directly; we copy the data in and out of + prev_cycle_state. */ + state_t prev_cycle_state_ctx; + int reg_n_accesses[FIRST_PSEUDO_REGISTER]; int reg_n_xaccesses[FIRST_PSEUDO_REGISTER]; int reg_set_in_cycle[FIRST_PSEUDO_REGISTER]; @@ -3109,6 +3207,11 @@ typedef struct c6x_sched_context /* The current scheduling state. */ static struct c6x_sched_context ss; +/* The following variable value is DFA state before issueing the first insn + in the current clock cycle. This is used in c6x_variable_issue for + comparison with the state after issuing the last insn in a cycle. */ +static state_t prev_cycle_state; + /* Set when we discover while processing an insn that it would lead to too many accesses of the same register. */ static bool reg_access_stall; @@ -3181,6 +3284,7 @@ insn_set_clock (rtx insn, int cycle) INSN_INFO_ENTRY (uid).clock = cycle; INSN_INFO_ENTRY (uid).new_cond = NULL; + INSN_INFO_ENTRY (uid).reservation = 0; INSN_INFO_ENTRY (uid).ebb_start = false; } @@ -3317,9 +3421,13 @@ init_sched_state (c6x_sched_context_t sc) sc->delays_finished_at = 0; sc->curr_sched_clock = 0; + sc->prev_cycle_state_ctx = xmalloc (dfa_state_size); + memset (sc->reg_n_accesses, 0, sizeof sc->reg_n_accesses); memset (sc->reg_n_xaccesses, 0, sizeof sc->reg_n_xaccesses); memset (sc->reg_set_in_cycle, 0, sizeof sc->reg_set_in_cycle); + + state_reset (sc->prev_cycle_state_ctx); } /* Allocate store for new scheduling context. */ @@ -3341,7 +3449,11 @@ c6x_init_sched_context (void *_sc, bool clean_p) init_sched_state (sc); } else - *sc = ss; + { + *sc = ss; + sc->prev_cycle_state_ctx = xmalloc (dfa_state_size); + memcpy (sc->prev_cycle_state_ctx, prev_cycle_state, dfa_state_size); + } } /* Sets the global scheduling context to the one pointed to by _SC. */ @@ -3352,6 +3464,17 @@ c6x_set_sched_context (void *_sc) gcc_assert (sc != NULL); ss = *sc; + memcpy (prev_cycle_state, sc->prev_cycle_state_ctx, dfa_state_size); +} + +/* Clear data in _SC. */ +static void +c6x_clear_sched_context (void *_sc) +{ + c6x_sched_context_t sc = (c6x_sched_context_t) _sc; + gcc_assert (_sc != NULL); + + free (sc->prev_cycle_state_ctx); } /* Free _SC. */ @@ -3384,6 +3507,17 @@ c6x_issue_rate (void) return 8; } +/* Used together with the collapse_ndfa option, this ensures that we reach a + deterministic automaton state before trying to advance a cycle. + With collapse_ndfa, genautomata creates advance cycle arcs only for + such deterministic states. */ + +static rtx +c6x_sched_dfa_pre_cycle_insn (void) +{ + return const0_rtx; +} + /* We're beginning a new block. Initialize data structures as necessary. */ static void @@ -3391,7 +3525,28 @@ c6x_sched_init (FILE *dump ATTRIBUTE_UNUSED, int sched_verbose ATTRIBUTE_UNUSED, int max_ready ATTRIBUTE_UNUSED) { + if (prev_cycle_state == NULL) + { + prev_cycle_state = xmalloc (dfa_state_size); + } init_sched_state (&ss); + state_reset (prev_cycle_state); +} + +/* We are about to being issuing INSN. Return nonzero if we cannot + issue it on given cycle CLOCK and return zero if we should not sort + the ready queue on the next clock start. + For C6X, we use this function just to copy the previous DFA state + for comparison purposes. */ + +static int +c6x_dfa_new_cycle (FILE *dump ATTRIBUTE_UNUSED, int verbose ATTRIBUTE_UNUSED, + rtx insn ATTRIBUTE_UNUSED, int last_clock ATTRIBUTE_UNUSED, + int clock ATTRIBUTE_UNUSED, int *sort_p ATTRIBUTE_UNUSED) +{ + if (clock != last_clock) + memcpy (prev_cycle_state, curr_state, dfa_state_size); + return 0; } static void @@ -3766,11 +3921,14 @@ c6x_variable_issue (FILE *dump ATTRIBUTE_UNUSED, ss.issued_this_cycle++; if (insn_info) { + state_t st_after = alloca (dfa_state_size); int curr_clock = ss.curr_sched_clock; int uid = INSN_UID (insn); int icode = recog_memoized (insn); rtx first_cond; int first, first_cycle; + unsigned int mask; + int i; insn_set_clock (insn, curr_clock); INSN_INFO_ENTRY (uid).ebb_start @@ -3795,6 +3953,16 @@ c6x_variable_issue (FILE *dump ATTRIBUTE_UNUSED, || get_attr_type (insn) == TYPE_CALL)) INSN_INFO_ENTRY (uid).new_cond = first_cond; + memcpy (st_after, curr_state, dfa_state_size); + state_transition (st_after, const0_rtx); + + mask = 0; + for (i = 0; i < 2 * UNIT_QID_SIDE_OFFSET; i++) + if (cpu_unit_reservation_p (st_after, c6x_unit_codes[i]) + && !cpu_unit_reservation_p (prev_cycle_state, c6x_unit_codes[i])) + mask |= 1 << i; + INSN_INFO_ENTRY (uid).unit_mask = mask; + maybe_clobber_cond (insn, curr_clock); if (icode >= 0) @@ -4323,15 +4491,19 @@ reorg_split_calls (rtx *call_labels) if (can_use_callp (insn)) { /* Find the first insn of the next execute packet. If it - is outside the branch delay slots of this call, we may + is the shadow insn corresponding to this call, we may use a CALLP insn. */ - rtx next_cycle_start = next_nonnote_nondebug_insn (last_same_clock); + rtx shadow = next_nonnote_nondebug_insn (last_same_clock); - if (CALL_P (next_cycle_start) - && (insn_get_clock (next_cycle_start) == this_clock + 5)) + if (CALL_P (shadow) + && insn_get_clock (shadow) == this_clock + 5) { - convert_to_callp (next_cycle_start); - insn_set_clock (next_cycle_start, this_clock); + convert_to_callp (shadow); + insn_set_clock (shadow, this_clock); + INSN_INFO_ENTRY (INSN_UID (shadow)).reservation + = RESERVATION_S2; + INSN_INFO_ENTRY (INSN_UID (shadow)).unit_mask + = INSN_INFO_ENTRY (INSN_UID (last_same_clock)).unit_mask; if (GET_MODE (insn) == TImode) { rtx new_cycle_first = NEXT_INSN (insn); @@ -4340,13 +4512,13 @@ reorg_split_calls (rtx *call_labels) || GET_CODE (PATTERN (new_cycle_first)) == CLOBBER) new_cycle_first = NEXT_INSN (new_cycle_first); PUT_MODE (new_cycle_first, TImode); - if (new_cycle_first != next_cycle_start) - PUT_MODE (next_cycle_start, VOIDmode); + if (new_cycle_first != shadow) + PUT_MODE (shadow, VOIDmode); INSN_INFO_ENTRY (INSN_UID (new_cycle_first)).ebb_start = INSN_INFO_ENTRY (INSN_UID (insn)).ebb_start; } else - PUT_MODE (next_cycle_start, VOIDmode); + PUT_MODE (shadow, VOIDmode); delete_insn (insn); goto done; } @@ -4364,6 +4536,9 @@ reorg_split_calls (rtx *call_labels) INSN_INFO_ENTRY (INSN_UID (x1)).reservation = RESERVATION_S2; if (after1 == last_same_clock) PUT_MODE (x1, TImode); + else + INSN_INFO_ENTRY (INSN_UID (x1)).unit_mask + = INSN_INFO_ENTRY (INSN_UID (after1)).unit_mask; } else { @@ -4381,8 +4556,14 @@ reorg_split_calls (rtx *call_labels) INSN_INFO_ENTRY (INSN_UID (x2)).reservation = RESERVATION_S2; if (after1 == last_same_clock) PUT_MODE (x1, TImode); + else + INSN_INFO_ENTRY (INSN_UID (x1)).unit_mask + = INSN_INFO_ENTRY (INSN_UID (after1)).unit_mask; if (after1 == after2) PUT_MODE (x2, TImode); + else + INSN_INFO_ENTRY (INSN_UID (x2)).unit_mask + = INSN_INFO_ENTRY (INSN_UID (after2)).unit_mask; } } } @@ -5524,6 +5705,10 @@ c6x_expand_builtin (tree exp, rtx target ATTRIBUTE_UNUSED, #define TARGET_SCHED_REORDER c6x_sched_reorder #undef TARGET_SCHED_REORDER2 #define TARGET_SCHED_REORDER2 c6x_sched_reorder2 +#undef TARGET_SCHED_DFA_NEW_CYCLE +#define TARGET_SCHED_DFA_NEW_CYCLE c6x_dfa_new_cycle +#undef TARGET_SCHED_DFA_PRE_CYCLE_INSN +#define TARGET_SCHED_DFA_PRE_CYCLE_INSN c6x_sched_dfa_pre_cycle_insn #undef TARGET_SCHED_EXPOSED_PIPELINE #define TARGET_SCHED_EXPOSED_PIPELINE true @@ -5533,6 +5718,8 @@ c6x_expand_builtin (tree exp, rtx target ATTRIBUTE_UNUSED, #define TARGET_SCHED_INIT_SCHED_CONTEXT c6x_init_sched_context #undef TARGET_SCHED_SET_SCHED_CONTEXT #define TARGET_SCHED_SET_SCHED_CONTEXT c6x_set_sched_context +#undef TARGET_SCHED_CLEAR_SCHED_CONTEXT +#define TARGET_SCHED_CLEAR_SCHED_CONTEXT c6x_clear_sched_context #undef TARGET_SCHED_FREE_SCHED_CONTEXT #define TARGET_SCHED_FREE_SCHED_CONTEXT c6x_free_sched_context diff --git a/gcc/config/c6x/c6x.h b/gcc/config/c6x/c6x.h index 5d34d590ff3..9b1b1636b52 100644 --- a/gcc/config/c6x/c6x.h +++ b/gcc/config/c6x/c6x.h @@ -612,6 +612,8 @@ do { \ #define Pmode SImode #define FUNCTION_MODE QImode +#define CPU_UNITS_QUERY 1 + extern int c6x_initial_flag_pic; #endif /* GCC_C6X_H */ diff --git a/gcc/config/c6x/c6x.md b/gcc/config/c6x/c6x.md index 4554cd08130..c2502ca1a42 100644 --- a/gcc/config/c6x/c6x.md +++ b/gcc/config/c6x/c6x.md @@ -242,21 +242,27 @@ ] (const_string "unknown"))) -(define_automaton "c6x_1,c6x_w1,c6x_2,c6x_w2,c6x_m1,c6x_m2,c6x_t1,c6x_t2,c6x_branch") +(define_automaton "c6x_1,c6x_2,c6x_m1,c6x_m2,c6x_t1,c6x_t2,c6x_branch") +(automata_option "no-comb-vect") (automata_option "ndfa") +(automata_option "collapse-ndfa") -(define_cpu_unit "d1,l1,s1" "c6x_1") +(define_query_cpu_unit "d1,l1,s1" "c6x_1") (define_cpu_unit "x1" "c6x_1") -(define_cpu_unit "l1w,s1w" "c6x_w1") -(define_cpu_unit "m1" "c6x_m1") +(define_cpu_unit "l1w,s1w" "c6x_1") +(define_query_cpu_unit "m1" "c6x_m1") (define_cpu_unit "m1w" "c6x_m1") (define_cpu_unit "t1" "c6x_t1") -(define_cpu_unit "d2,l2,s2" "c6x_2") +(define_query_cpu_unit "d2,l2,s2" "c6x_2") (define_cpu_unit "x2" "c6x_2") -(define_cpu_unit "l2w,s2w" "c6x_w2") -(define_cpu_unit "m2" "c6x_m2") +(define_cpu_unit "l2w,s2w" "c6x_2") +(define_query_cpu_unit "m2" "c6x_m2") (define_cpu_unit "m2w" "c6x_m2") (define_cpu_unit "t2" "c6x_t2") +;; A special set of units used to identify specific reservations, rather than +;; just units. +(define_query_cpu_unit "fps1,fpl1,adddps1,adddpl1" "c6x_1") +(define_query_cpu_unit "fps2,fpl2,adddps2,adddpl2" "c6x_2") ;; There can be up to two branches in one cycle (on the .s1 and .s2 ;; units), but some instructions must not be scheduled in parallel |