]> granicus.if.org Git - postgresql/commitdiff
Avoid possibly accessing off the end of memory in examine_attribute().
authorTom Lane <tgl@sss.pgh.pa.us>
Tue, 6 Sep 2011 18:35:55 +0000 (14:35 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Tue, 6 Sep 2011 18:38:11 +0000 (14:38 -0400)
Since the last couple of columns of pg_type are often NULL,
sizeof(FormData_pg_type) can be an overestimate of the actual size of the
tuple data part.  Therefore memcpy'ing that much out of the catalog cache,
as analyze.c was doing, poses a small risk of copying past the end of
memory and incurring SIGSEGV.  No such crash has been identified in the
field, but we've certainly seen the equivalent happen in other code paths,
so patch this one all the way back.

Per valgrind testing by Noah Misch, though this is not his proposed patch.
I chose to use SearchSysCacheCopy1 rather than inventing special-purpose
infrastructure for copying only the minimal part of a pg_type tuple.

src/backend/commands/analyze.c

index 5c7cc650e3e8c51648e1b48fcb38885143e13231..c730cfc77919defc18651300f9d020d0c70eafe2 100644 (file)
@@ -702,14 +702,12 @@ examine_attribute(Relation onerel, int attnum)
        stats = (VacAttrStats *) palloc0(sizeof(VacAttrStats));
        stats->attr = (Form_pg_attribute) palloc(ATTRIBUTE_TUPLE_SIZE);
        memcpy(stats->attr, attr, ATTRIBUTE_TUPLE_SIZE);
-       typtuple = SearchSysCache(TYPEOID,
-                                                         ObjectIdGetDatum(attr->atttypid),
-                                                         0, 0, 0);
+       typtuple = SearchSysCacheCopy(TYPEOID,
+                                                                 ObjectIdGetDatum(attr->atttypid),
+                                                                 0, 0, 0);
        if (!HeapTupleIsValid(typtuple))
                elog(ERROR, "cache lookup failed for type %u", attr->atttypid);
-       stats->attrtype = (Form_pg_type) palloc(sizeof(FormData_pg_type));
-       memcpy(stats->attrtype, GETSTRUCT(typtuple), sizeof(FormData_pg_type));
-       ReleaseSysCache(typtuple);
+       stats->attrtype = (Form_pg_type) GETSTRUCT(typtuple);
        stats->anl_context = anl_context;
        stats->tupattnum = attnum;
 
@@ -725,7 +723,7 @@ examine_attribute(Relation onerel, int attnum)
 
        if (!ok || stats->compute_stats == NULL || stats->minrows <= 0)
        {
-               pfree(stats->attrtype);
+               heap_freetuple(typtuple);
                pfree(stats->attr);
                pfree(stats);
                return NULL;