2
0
mirror of git://gcc.gnu.org/git/gcc.git synced 2025-04-03 08:50:31 +08:00

Avoid infinite recursion looking up method in invalid recursive type.

From-SVN: r168186
This commit is contained in:
Ian Lance Taylor 2010-12-22 23:48:08 +00:00
parent 7cfc62ba9e
commit abff6b5fb5
2 changed files with 37 additions and 12 deletions
gcc/go/gofrontend

@ -7622,13 +7622,14 @@ Type::bind_field_or_method(Gogo* gogo, const Type* type, Expression* expr,
bool receiver_can_be_pointer = (expr->type()->points_to() != NULL
|| expr->is_addressable());
std::vector<const Named_type*> seen;
bool is_method = false;
bool found_pointer_method = false;
std::string ambig1;
std::string ambig2;
if (Type::find_field_or_method(type, name, receiver_can_be_pointer, NULL,
&is_method, &found_pointer_method,
&ambig1, &ambig2))
if (Type::find_field_or_method(type, name, receiver_can_be_pointer,
&seen, NULL, &is_method,
&found_pointer_method, &ambig1, &ambig2))
{
Expression* ret;
if (!is_method)
@ -7704,13 +7705,16 @@ Type::bind_field_or_method(Gogo* gogo, const Type* type, Expression* expr,
// ambiguity. If a method is found, sets *IS_METHOD to true;
// otherwise, if a field is found, set it to false. If
// RECEIVER_CAN_BE_POINTER is false, then the receiver is a value
// whose address can not be taken. When returning false, this sets
// *FOUND_POINTER_METHOD if we found a method we couldn't use because
// it requires a pointer. LEVEL is used for recursive calls, and can
// be NULL for a non-recursive call. When this function returns false
// because it finds that the name is ambiguous, it will store a path
// to the ambiguous names in *AMBIG1 and *AMBIG2. If the name is not
// found at all, *AMBIG1 and *AMBIG2 will be unchanged.
// whose address can not be taken. SEEN is used to avoid infinite
// recursion on invalid types.
// When returning false, this sets *FOUND_POINTER_METHOD if we found a
// method we couldn't use because it requires a pointer. LEVEL is
// used for recursive calls, and can be NULL for a non-recursive call.
// When this function returns false because it finds that the name is
// ambiguous, it will store a path to the ambiguous names in *AMBIG1
// and *AMBIG2. If the name is not found at all, *AMBIG1 and *AMBIG2
// will be unchanged.
// This function just returns whether or not there is a field or
// method, and whether it is a field or method. It doesn't build an
@ -7723,6 +7727,7 @@ bool
Type::find_field_or_method(const Type* type,
const std::string& name,
bool receiver_can_be_pointer,
std::vector<const Named_type*>* seen,
int* level,
bool* is_method,
bool* found_pointer_method,
@ -7749,6 +7754,17 @@ Type::find_field_or_method(const Type* type,
// else.
*found_pointer_method = true;
}
for (std::vector<const Named_type*>::const_iterator p = seen->begin();
p != seen->end();
++p)
{
if (*p == nt)
{
// We've already seen this type when searching for methods.
return false;
}
}
}
// Interface types can have methods.
@ -7768,6 +7784,9 @@ Type::find_field_or_method(const Type* type,
if (fields == NULL)
return false;
if (nt != NULL)
seen->push_back(nt);
int found_level = 0;
bool found_is_method = false;
std::string found_ambig1;
@ -7780,6 +7799,8 @@ Type::find_field_or_method(const Type* type,
if (pf->field_name() == name)
{
*is_method = false;
if (nt != NULL)
seen->pop_back();
return true;
}
@ -7800,6 +7821,7 @@ Type::find_field_or_method(const Type* type,
bool subfound = Type::find_field_or_method(fnt,
name,
receiver_can_be_pointer,
seen,
&sublevel,
&sub_is_method,
found_pointer_method,
@ -7856,6 +7878,9 @@ Type::find_field_or_method(const Type* type,
// FOUND_AMBIG2 are not empty. If we found the field, FOUND_LEVEL
// is not 0 and FOUND_AMBIG1 and FOUND_AMBIG2 are empty.
if (nt != NULL)
seen->pop_back();
if (found_level == 0)
return false;
else if (!found_ambig1.empty())

@ -1067,8 +1067,8 @@ class Type
static bool
find_field_or_method(const Type* type, const std::string& name,
bool receiver_can_be_pointer,
int* level, bool* is_method,
bool* found_pointer_method,
std::vector<const Named_type*>*, int* level,
bool* is_method, bool* found_pointer_method,
std::string* ambig1, std::string* ambig2);
// Get a tree for a type without looking in the hash table for