diff options
author | sayle <sayle@138bc75d-0d04-0410-961f-82ee72b054a4> | 2003-12-02 01:39:20 +0000 |
---|---|---|
committer | sayle <sayle@138bc75d-0d04-0410-961f-82ee72b054a4> | 2003-12-02 01:39:20 +0000 |
commit | 6933395236ae08dad201096d3572e9a1fb29a0bc (patch) | |
tree | 9af4e807352e25d03e2723102e67712a53d561d9 | |
parent | 341e6a1b1c322ec6818a6abfdac8a5234f713fa1 (diff) | |
download | gcc-6933395236ae08dad201096d3572e9a1fb29a0bc.tar.gz |
PR optimization/12322
* gcse.c (struct ls_expr): Change type of hash_index from int to
unsigned int.
(hash_expr): Document hash_table_size parameter and wrap long line.
(ldst_entry): Calculate expression's hash_index and record in ptr.
(trim_ld_motion_mems): Use hash_index to search a single bucket
instead of scanning the entire hash_table. Remove the "del" local
variable and use the equivalent "expr == 0" instead. Change last
to be a pointer to the pointer to the current element, to simplify
and speed-up deleting from a linked list.
git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@74144 138bc75d-0d04-0410-961f-82ee72b054a4
-rw-r--r-- | gcc/ChangeLog | 13 | ||||
-rw-r--r-- | gcc/gcse.c | 100 |
2 files changed, 57 insertions, 56 deletions
diff --git a/gcc/ChangeLog b/gcc/ChangeLog index 47096ff677f..3f755a66190 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,3 +1,16 @@ +2003-12-01 Roger Sayle <roger@eyesopen.com> + + PR optimization/12322 + * gcse.c (struct ls_expr): Change type of hash_index from int to + unsigned int. + (hash_expr): Document hash_table_size parameter and wrap long line. + (ldst_entry): Calculate expression's hash_index and record in ptr. + (trim_ld_motion_mems): Use hash_index to search a single bucket + instead of scanning the entire hash_table. Remove the "del" local + variable and use the equivalent "expr == 0" instead. Change last + to be a pointer to the pointer to the current element, to simplify + and speed-up deleting from a linked list. + 2003-12-01 James E Wilson <wilson@specifixinc.com> * doc/c-tree.texi (CONSTRUCTOR): Clarify element order and handling diff --git a/gcc/gcse.c b/gcc/gcse.c index 8f9f13aa4cb..73f293bf24e 100644 --- a/gcc/gcse.c +++ b/gcc/gcse.c @@ -468,7 +468,7 @@ struct ls_expr struct ls_expr * next; /* Next in the list. */ int invalid; /* Invalid for some reason. */ int index; /* If it maps to a bitmap index. */ - int hash_index; /* Index when in a hash table. */ + unsigned int hash_index; /* Index when in a hash table. */ rtx reaching_reg; /* Register to use when re-writing. */ }; @@ -1513,12 +1513,14 @@ oprs_available_p (rtx x, rtx insn) MODE is only used if X is a CONST_INT. DO_NOT_RECORD_P is a boolean indicating if a volatile operand is found or if the expression contains - something we don't want to insert in the table. + something we don't want to insert in the table. HASH_TABLE_SIZE is + the current size of the hash table to be probed. ??? One might want to merge this with canon_hash. Later. */ static unsigned int -hash_expr (rtx x, enum machine_mode mode, int *do_not_record_p, int hash_table_size) +hash_expr (rtx x, enum machine_mode mode, int *do_not_record_p, + int hash_table_size) { unsigned int hash; @@ -6519,28 +6521,29 @@ one_code_hoisting_pass (void) static struct ls_expr * ldst_entry (rtx x) { + int do_not_record_p = 0; struct ls_expr * ptr; + unsigned int hash; - for (ptr = first_ls_expr(); ptr != NULL; ptr = next_ls_expr (ptr)) - if (expr_equiv_p (ptr->pattern, x)) - break; + hash = hash_expr_1 (x, GET_MODE (x), & do_not_record_p); - if (!ptr) - { - ptr = xmalloc (sizeof (struct ls_expr)); + for (ptr = pre_ldst_mems; ptr != NULL; ptr = ptr->next) + if (ptr->hash_index == hash && expr_equiv_p (ptr->pattern, x)) + return ptr; - ptr->next = pre_ldst_mems; - ptr->expr = NULL; - ptr->pattern = x; - ptr->pattern_regs = NULL_RTX; - ptr->loads = NULL_RTX; - ptr->stores = NULL_RTX; - ptr->reaching_reg = NULL_RTX; - ptr->invalid = 0; - ptr->index = 0; - ptr->hash_index = 0; - pre_ldst_mems = ptr; - } + ptr = xmalloc (sizeof (struct ls_expr)); + + ptr->next = pre_ldst_mems; + ptr->expr = NULL; + ptr->pattern = x; + ptr->pattern_regs = NULL_RTX; + ptr->loads = NULL_RTX; + ptr->stores = NULL_RTX; + ptr->reaching_reg = NULL_RTX; + ptr->invalid = 0; + ptr->index = 0; + ptr->hash_index = hash; + pre_ldst_mems = ptr; return ptr; } @@ -6800,56 +6803,41 @@ compute_ld_motion_mems (void) static void trim_ld_motion_mems (void) { - struct ls_expr * last = NULL; - struct ls_expr * ptr = first_ls_expr (); + struct ls_expr * * last = & pre_ldst_mems; + struct ls_expr * ptr = pre_ldst_mems; while (ptr != NULL) { - int del = ptr->invalid; - struct expr * expr = NULL; + struct expr * expr; /* Delete if entry has been made invalid. */ - if (!del) + if (! ptr->invalid) { - unsigned int i; - - del = 1; /* Delete if we cannot find this mem in the expression list. */ - for (i = 0; i < expr_hash_table.size && del; i++) - { - for (expr = expr_hash_table.table[i]; - expr != NULL; - expr = expr->next_same_hash) - if (expr_equiv_p (expr->expr, ptr->pattern)) - { - del = 0; - break; - } - } - } + unsigned int hash = ptr->hash_index % expr_hash_table.size; - if (del) - { - if (last != NULL) - { - last->next = ptr->next; - free_ldst_entry (ptr); - ptr = last->next; - } - else - { - pre_ldst_mems = pre_ldst_mems->next; - free_ldst_entry (ptr); - ptr = pre_ldst_mems; - } + for (expr = expr_hash_table.table[hash]; + expr != NULL; + expr = expr->next_same_hash) + if (expr_equiv_p (expr->expr, ptr->pattern)) + break; } else + expr = (struct expr *) 0; + + if (expr) { /* Set the expression field if we are keeping it. */ - last = ptr; ptr->expr = expr; + last = & ptr->next; ptr = ptr->next; } + else + { + *last = ptr->next; + free_ldst_entry (ptr); + ptr = * last; + } } /* Show the world what we've found. */ |