Location via proxy:   [ UP ]  
[Report a bug]   [Manage cookies]                
Clear padding of PgStat_HashKey when handling pgstats entries
authorMichael Paquier <michael@paquier.xyz>
Tue, 5 Nov 2024 00:41:01 +0000 (09:41 +0900)
committerMichael Paquier <michael@paquier.xyz>
Tue, 5 Nov 2024 00:41:01 +0000 (09:41 +0900)
PgStat_HashKey is currently initialized in a way that could result in
random data if the structure has any padding bytes.  The structure
has no padding bytes currently, fortunately, but it could become a
problem should the structure change at some point in the future.

The code is changed to use some memset(0) so as any padding would be
handled properly, as it would be surprising to see random failures in
the pgstats entry lookups.  PgStat_HashKey is a structure internal to
pgstats, and an ABI change could be possible in the scope of a bug fix,
so backpatch down to 15 where this has been introduced.

Author: Bertrand Drouvot
Reviewed-by: Jelte Fennema-Nio, Michael Paquier
Discussion: https://postgr.es/m/Zyb7RW1y9dVfO0UH@ip-10-97-1-34.eu-west-3.compute.internal
Backpatch-through: 15

src/backend/utils/activity/pgstat.c
src/backend/utils/activity/pgstat_shmem.c

index 142e26af52e51618ea81fba25c3a6e8bb13548f6..15f5f2b714164a1e0c10041816e271560e64c566 100644 (file)
@@ -800,6 +800,9 @@ pgstat_fetch_entry(PgStat_Kind kind, Oid dboid, Oid objoid)
 
    pgstat_prep_snapshot();
 
+   /* clear padding */
+   memset(&key, 0, sizeof(struct PgStat_HashKey));
+
    key.kind = kind;
    key.dboid = dboid;
    key.objoid = objoid;
index 2b6c7a07f88dec2e8aa775b95e15036f2c901e8d..2b3e7c491a970aacc8b4bc0fa3cfb705a27fca1c 100644 (file)
@@ -401,11 +401,18 @@ PgStat_EntryRef *
 pgstat_get_entry_ref(PgStat_Kind kind, Oid dboid, Oid objoid, bool create,
                     bool *created_entry)
 {
-   PgStat_HashKey key = {.kind = kind,.dboid = dboid,.objoid = objoid};
+   PgStat_HashKey key;
    PgStatShared_HashEntry *shhashent;
    PgStatShared_Common *shheader = NULL;
    PgStat_EntryRef *entry_ref;
 
+   /* clear padding */
+   memset(&key, 0, sizeof(struct PgStat_HashKey));
+
+   key.kind = kind;
+   key.dboid = dboid;
+   key.objoid = objoid;
+
    /*
     * passing in created_entry only makes sense if we possibly could create
     * entry.
@@ -876,10 +883,17 @@ pgstat_drop_database_and_contents(Oid dboid)
 bool
 pgstat_drop_entry(PgStat_Kind kind, Oid dboid, Oid objoid)
 {
-   PgStat_HashKey key = {.kind = kind,.dboid = dboid,.objoid = objoid};
+   PgStat_HashKey key;
    PgStatShared_HashEntry *shent;
    bool        freed = true;
 
+   /* clear padding */
+   memset(&key, 0, sizeof(struct PgStat_HashKey));
+
+   key.kind = kind;
+   key.dboid = dboid;
+   key.objoid = objoid;
+
    /* delete local reference */
    if (pgStatEntryRefHash)
    {