LCOV - code coverage report
Current view: top level - src/backend/commands - seclabel.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 124 176 70.5 %
Date: 2025-01-18 04:15:08 Functions: 7 9 77.8 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -------------------------------------------------------------------------
       2             :  *
       3             :  * seclabel.c
       4             :  *    routines to support security label feature.
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  * -------------------------------------------------------------------------
      10             :  */
      11             : #include "postgres.h"
      12             : 
      13             : #include "access/genam.h"
      14             : #include "access/htup_details.h"
      15             : #include "access/relation.h"
      16             : #include "access/table.h"
      17             : #include "catalog/catalog.h"
      18             : #include "catalog/indexing.h"
      19             : #include "catalog/pg_seclabel.h"
      20             : #include "catalog/pg_shseclabel.h"
      21             : #include "commands/seclabel.h"
      22             : #include "miscadmin.h"
      23             : #include "utils/builtins.h"
      24             : #include "utils/fmgroids.h"
      25             : #include "utils/memutils.h"
      26             : #include "utils/rel.h"
      27             : 
      28             : typedef struct
      29             : {
      30             :     const char *provider_name;
      31             :     check_object_relabel_type hook;
      32             : } LabelProvider;
      33             : 
      34             : static List *label_provider_list = NIL;
      35             : 
      36             : static bool
      37          48 : SecLabelSupportsObjectType(ObjectType objtype)
      38             : {
      39          48 :     switch (objtype)
      40             :     {
      41          48 :         case OBJECT_AGGREGATE:
      42             :         case OBJECT_COLUMN:
      43             :         case OBJECT_DATABASE:
      44             :         case OBJECT_DOMAIN:
      45             :         case OBJECT_EVENT_TRIGGER:
      46             :         case OBJECT_FOREIGN_TABLE:
      47             :         case OBJECT_FUNCTION:
      48             :         case OBJECT_LANGUAGE:
      49             :         case OBJECT_LARGEOBJECT:
      50             :         case OBJECT_MATVIEW:
      51             :         case OBJECT_PROCEDURE:
      52             :         case OBJECT_PUBLICATION:
      53             :         case OBJECT_ROLE:
      54             :         case OBJECT_ROUTINE:
      55             :         case OBJECT_SCHEMA:
      56             :         case OBJECT_SEQUENCE:
      57             :         case OBJECT_SUBSCRIPTION:
      58             :         case OBJECT_TABLE:
      59             :         case OBJECT_TABLESPACE:
      60             :         case OBJECT_TYPE:
      61             :         case OBJECT_VIEW:
      62          48 :             return true;
      63             : 
      64           0 :         case OBJECT_ACCESS_METHOD:
      65             :         case OBJECT_AMOP:
      66             :         case OBJECT_AMPROC:
      67             :         case OBJECT_ATTRIBUTE:
      68             :         case OBJECT_CAST:
      69             :         case OBJECT_COLLATION:
      70             :         case OBJECT_CONVERSION:
      71             :         case OBJECT_DEFAULT:
      72             :         case OBJECT_DEFACL:
      73             :         case OBJECT_DOMCONSTRAINT:
      74             :         case OBJECT_EXTENSION:
      75             :         case OBJECT_FDW:
      76             :         case OBJECT_FOREIGN_SERVER:
      77             :         case OBJECT_INDEX:
      78             :         case OBJECT_OPCLASS:
      79             :         case OBJECT_OPERATOR:
      80             :         case OBJECT_OPFAMILY:
      81             :         case OBJECT_PARAMETER_ACL:
      82             :         case OBJECT_POLICY:
      83             :         case OBJECT_PUBLICATION_NAMESPACE:
      84             :         case OBJECT_PUBLICATION_REL:
      85             :         case OBJECT_RULE:
      86             :         case OBJECT_STATISTIC_EXT:
      87             :         case OBJECT_TABCONSTRAINT:
      88             :         case OBJECT_TRANSFORM:
      89             :         case OBJECT_TRIGGER:
      90             :         case OBJECT_TSCONFIGURATION:
      91             :         case OBJECT_TSDICTIONARY:
      92             :         case OBJECT_TSPARSER:
      93             :         case OBJECT_TSTEMPLATE:
      94             :         case OBJECT_USER_MAPPING:
      95           0 :             return false;
      96             : 
      97             :             /*
      98             :              * There's intentionally no default: case here; we want the
      99             :              * compiler to warn if a new ObjectType hasn't been handled above.
     100             :              */
     101             :     }
     102             : 
     103             :     /* Shouldn't get here, but if we do, say "no support" */
     104           0 :     return false;
     105             : }
     106             : 
     107             : /*
     108             :  * ExecSecLabelStmt --
     109             :  *
     110             :  * Apply a security label to a database object.
     111             :  *
     112             :  * Returns the ObjectAddress of the object to which the policy was applied.
     113             :  */
     114             : ObjectAddress
     115         100 : ExecSecLabelStmt(SecLabelStmt *stmt)
     116             : {
     117         100 :     LabelProvider *provider = NULL;
     118             :     ObjectAddress address;
     119             :     Relation    relation;
     120             :     ListCell   *lc;
     121             : 
     122             :     /*
     123             :      * Find the named label provider, or if none specified, check whether
     124             :      * there's exactly one, and if so use it.
     125             :      */
     126         100 :     if (stmt->provider == NULL)
     127             :     {
     128          80 :         if (label_provider_list == NIL)
     129          36 :             ereport(ERROR,
     130             :                     (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     131             :                      errmsg("no security label providers have been loaded")));
     132          44 :         if (list_length(label_provider_list) != 1)
     133           0 :             ereport(ERROR,
     134             :                     (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     135             :                      errmsg("must specify provider when multiple security label providers have been loaded")));
     136          44 :         provider = (LabelProvider *) linitial(label_provider_list);
     137             :     }
     138             :     else
     139             :     {
     140          24 :         foreach(lc, label_provider_list)
     141             :         {
     142           8 :             LabelProvider *lp = lfirst(lc);
     143             : 
     144           8 :             if (strcmp(stmt->provider, lp->provider_name) == 0)
     145             :             {
     146           4 :                 provider = lp;
     147           4 :                 break;
     148             :             }
     149             :         }
     150          20 :         if (provider == NULL)
     151          16 :             ereport(ERROR,
     152             :                     (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     153             :                      errmsg("security label provider \"%s\" is not loaded",
     154             :                             stmt->provider)));
     155             :     }
     156             : 
     157          48 :     if (!SecLabelSupportsObjectType(stmt->objtype))
     158           0 :         ereport(ERROR,
     159             :                 (errcode(ERRCODE_WRONG_OBJECT_TYPE),
     160             :                  errmsg("security labels are not supported for this type of object")));
     161             : 
     162             :     /*
     163             :      * Translate the parser representation which identifies this object into
     164             :      * an ObjectAddress. get_object_address() will throw an error if the
     165             :      * object does not exist, and will also acquire a lock on the target to
     166             :      * guard against concurrent modifications.
     167             :      */
     168          48 :     address = get_object_address(stmt->objtype, stmt->object,
     169             :                                  &relation, ShareUpdateExclusiveLock, false);
     170             : 
     171             :     /* Require ownership of the target object. */
     172          42 :     check_object_ownership(GetUserId(), stmt->objtype, address,
     173             :                            stmt->object, relation);
     174             : 
     175             :     /* Perform other integrity checks as needed. */
     176          36 :     switch (stmt->objtype)
     177             :     {
     178           2 :         case OBJECT_COLUMN:
     179             : 
     180             :             /*
     181             :              * Allow security labels only on columns of tables, views,
     182             :              * materialized views, composite types, and foreign tables (which
     183             :              * are the only relkinds for which pg_dump will dump labels).
     184             :              */
     185           2 :             if (relation->rd_rel->relkind != RELKIND_RELATION &&
     186           0 :                 relation->rd_rel->relkind != RELKIND_VIEW &&
     187           0 :                 relation->rd_rel->relkind != RELKIND_MATVIEW &&
     188           0 :                 relation->rd_rel->relkind != RELKIND_COMPOSITE_TYPE &&
     189           0 :                 relation->rd_rel->relkind != RELKIND_FOREIGN_TABLE &&
     190           0 :                 relation->rd_rel->relkind != RELKIND_PARTITIONED_TABLE)
     191           0 :                 ereport(ERROR,
     192             :                         (errcode(ERRCODE_WRONG_OBJECT_TYPE),
     193             :                          errmsg("cannot set security label on relation \"%s\"",
     194             :                                 RelationGetRelationName(relation)),
     195             :                          errdetail_relkind_not_supported(relation->rd_rel->relkind)));
     196           2 :             break;
     197          34 :         default:
     198          34 :             break;
     199             :     }
     200             : 
     201             :     /* Provider gets control here, may throw ERROR to veto new label. */
     202          36 :     provider->hook(&address, stmt->label);
     203             : 
     204             :     /* Apply new label. */
     205          28 :     SetSecurityLabel(&address, provider->provider_name, stmt->label);
     206             : 
     207             :     /*
     208             :      * If get_object_address() opened the relation for us, we close it to keep
     209             :      * the reference count correct - but we retain any locks acquired by
     210             :      * get_object_address() until commit time, to guard against concurrent
     211             :      * activity.
     212             :      */
     213          28 :     if (relation != NULL)
     214          14 :         relation_close(relation, NoLock);
     215             : 
     216          28 :     return address;
     217             : }
     218             : 
     219             : /*
     220             :  * GetSharedSecurityLabel returns the security label for a shared object for
     221             :  * a given provider, or NULL if there is no such label.
     222             :  */
     223             : static char *
     224           0 : GetSharedSecurityLabel(const ObjectAddress *object, const char *provider)
     225             : {
     226             :     Relation    pg_shseclabel;
     227             :     ScanKeyData keys[3];
     228             :     SysScanDesc scan;
     229             :     HeapTuple   tuple;
     230             :     Datum       datum;
     231             :     bool        isnull;
     232           0 :     char       *seclabel = NULL;
     233             : 
     234           0 :     ScanKeyInit(&keys[0],
     235             :                 Anum_pg_shseclabel_objoid,
     236             :                 BTEqualStrategyNumber, F_OIDEQ,
     237             :                 ObjectIdGetDatum(object->objectId));
     238           0 :     ScanKeyInit(&keys[1],
     239             :                 Anum_pg_shseclabel_classoid,
     240             :                 BTEqualStrategyNumber, F_OIDEQ,
     241             :                 ObjectIdGetDatum(object->classId));
     242           0 :     ScanKeyInit(&keys[2],
     243             :                 Anum_pg_shseclabel_provider,
     244             :                 BTEqualStrategyNumber, F_TEXTEQ,
     245           0 :                 CStringGetTextDatum(provider));
     246             : 
     247           0 :     pg_shseclabel = table_open(SharedSecLabelRelationId, AccessShareLock);
     248             : 
     249           0 :     scan = systable_beginscan(pg_shseclabel, SharedSecLabelObjectIndexId,
     250             :                               criticalSharedRelcachesBuilt, NULL, 3, keys);
     251             : 
     252           0 :     tuple = systable_getnext(scan);
     253           0 :     if (HeapTupleIsValid(tuple))
     254             :     {
     255           0 :         datum = heap_getattr(tuple, Anum_pg_shseclabel_label,
     256             :                              RelationGetDescr(pg_shseclabel), &isnull);
     257           0 :         if (!isnull)
     258           0 :             seclabel = TextDatumGetCString(datum);
     259             :     }
     260           0 :     systable_endscan(scan);
     261             : 
     262           0 :     table_close(pg_shseclabel, AccessShareLock);
     263             : 
     264           0 :     return seclabel;
     265             : }
     266             : 
     267             : /*
     268             :  * GetSecurityLabel returns the security label for a shared or database object
     269             :  * for a given provider, or NULL if there is no such label.
     270             :  */
     271             : char *
     272           0 : GetSecurityLabel(const ObjectAddress *object, const char *provider)
     273             : {
     274             :     Relation    pg_seclabel;
     275             :     ScanKeyData keys[4];
     276             :     SysScanDesc scan;
     277             :     HeapTuple   tuple;
     278             :     Datum       datum;
     279             :     bool        isnull;
     280           0 :     char       *seclabel = NULL;
     281             : 
     282             :     /* Shared objects have their own security label catalog. */
     283           0 :     if (IsSharedRelation(object->classId))
     284           0 :         return GetSharedSecurityLabel(object, provider);
     285             : 
     286             :     /* Must be an unshared object, so examine pg_seclabel. */
     287           0 :     ScanKeyInit(&keys[0],
     288             :                 Anum_pg_seclabel_objoid,
     289             :                 BTEqualStrategyNumber, F_OIDEQ,
     290             :                 ObjectIdGetDatum(object->objectId));
     291           0 :     ScanKeyInit(&keys[1],
     292             :                 Anum_pg_seclabel_classoid,
     293             :                 BTEqualStrategyNumber, F_OIDEQ,
     294             :                 ObjectIdGetDatum(object->classId));
     295           0 :     ScanKeyInit(&keys[2],
     296             :                 Anum_pg_seclabel_objsubid,
     297             :                 BTEqualStrategyNumber, F_INT4EQ,
     298             :                 Int32GetDatum(object->objectSubId));
     299           0 :     ScanKeyInit(&keys[3],
     300             :                 Anum_pg_seclabel_provider,
     301             :                 BTEqualStrategyNumber, F_TEXTEQ,
     302           0 :                 CStringGetTextDatum(provider));
     303             : 
     304           0 :     pg_seclabel = table_open(SecLabelRelationId, AccessShareLock);
     305             : 
     306           0 :     scan = systable_beginscan(pg_seclabel, SecLabelObjectIndexId, true,
     307             :                               NULL, 4, keys);
     308             : 
     309           0 :     tuple = systable_getnext(scan);
     310           0 :     if (HeapTupleIsValid(tuple))
     311             :     {
     312           0 :         datum = heap_getattr(tuple, Anum_pg_seclabel_label,
     313             :                              RelationGetDescr(pg_seclabel), &isnull);
     314           0 :         if (!isnull)
     315           0 :             seclabel = TextDatumGetCString(datum);
     316             :     }
     317           0 :     systable_endscan(scan);
     318             : 
     319           0 :     table_close(pg_seclabel, AccessShareLock);
     320             : 
     321           0 :     return seclabel;
     322             : }
     323             : 
     324             : /*
     325             :  * SetSharedSecurityLabel is a helper function of SetSecurityLabel to
     326             :  * handle shared database objects.
     327             :  */
     328             : static void
     329           6 : SetSharedSecurityLabel(const ObjectAddress *object,
     330             :                        const char *provider, const char *label)
     331             : {
     332             :     Relation    pg_shseclabel;
     333             :     ScanKeyData keys[4];
     334             :     SysScanDesc scan;
     335             :     HeapTuple   oldtup;
     336           6 :     HeapTuple   newtup = NULL;
     337             :     Datum       values[Natts_pg_shseclabel];
     338             :     bool        nulls[Natts_pg_shseclabel];
     339             :     bool        replaces[Natts_pg_shseclabel];
     340             : 
     341             :     /* Prepare to form or update a tuple, if necessary. */
     342           6 :     memset(nulls, false, sizeof(nulls));
     343           6 :     memset(replaces, false, sizeof(replaces));
     344           6 :     values[Anum_pg_shseclabel_objoid - 1] = ObjectIdGetDatum(object->objectId);
     345           6 :     values[Anum_pg_shseclabel_classoid - 1] = ObjectIdGetDatum(object->classId);
     346           6 :     values[Anum_pg_shseclabel_provider - 1] = CStringGetTextDatum(provider);
     347           6 :     if (label != NULL)
     348           6 :         values[Anum_pg_shseclabel_label - 1] = CStringGetTextDatum(label);
     349             : 
     350             :     /* Use the index to search for a matching old tuple */
     351           6 :     ScanKeyInit(&keys[0],
     352             :                 Anum_pg_shseclabel_objoid,
     353             :                 BTEqualStrategyNumber, F_OIDEQ,
     354             :                 ObjectIdGetDatum(object->objectId));
     355           6 :     ScanKeyInit(&keys[1],
     356             :                 Anum_pg_shseclabel_classoid,
     357             :                 BTEqualStrategyNumber, F_OIDEQ,
     358             :                 ObjectIdGetDatum(object->classId));
     359           6 :     ScanKeyInit(&keys[2],
     360             :                 Anum_pg_shseclabel_provider,
     361             :                 BTEqualStrategyNumber, F_TEXTEQ,
     362           6 :                 CStringGetTextDatum(provider));
     363             : 
     364           6 :     pg_shseclabel = table_open(SharedSecLabelRelationId, RowExclusiveLock);
     365             : 
     366           6 :     scan = systable_beginscan(pg_shseclabel, SharedSecLabelObjectIndexId, true,
     367             :                               NULL, 3, keys);
     368             : 
     369           6 :     oldtup = systable_getnext(scan);
     370           6 :     if (HeapTupleIsValid(oldtup))
     371             :     {
     372           0 :         if (label == NULL)
     373           0 :             CatalogTupleDelete(pg_shseclabel, &oldtup->t_self);
     374             :         else
     375             :         {
     376           0 :             replaces[Anum_pg_shseclabel_label - 1] = true;
     377           0 :             newtup = heap_modify_tuple(oldtup, RelationGetDescr(pg_shseclabel),
     378             :                                        values, nulls, replaces);
     379           0 :             CatalogTupleUpdate(pg_shseclabel, &oldtup->t_self, newtup);
     380             :         }
     381             :     }
     382           6 :     systable_endscan(scan);
     383             : 
     384             :     /* If we didn't find an old tuple, insert a new one */
     385           6 :     if (newtup == NULL && label != NULL)
     386             :     {
     387           6 :         newtup = heap_form_tuple(RelationGetDescr(pg_shseclabel),
     388             :                                  values, nulls);
     389           6 :         CatalogTupleInsert(pg_shseclabel, newtup);
     390             :     }
     391             : 
     392           6 :     if (newtup != NULL)
     393           6 :         heap_freetuple(newtup);
     394             : 
     395           6 :     table_close(pg_shseclabel, RowExclusiveLock);
     396           6 : }
     397             : 
     398             : /*
     399             :  * SetSecurityLabel attempts to set the security label for the specified
     400             :  * provider on the specified object to the given value.  NULL means that any
     401             :  * existing label should be deleted.
     402             :  */
     403             : void
     404          28 : SetSecurityLabel(const ObjectAddress *object,
     405             :                  const char *provider, const char *label)
     406             : {
     407             :     Relation    pg_seclabel;
     408             :     ScanKeyData keys[4];
     409             :     SysScanDesc scan;
     410             :     HeapTuple   oldtup;
     411          28 :     HeapTuple   newtup = NULL;
     412             :     Datum       values[Natts_pg_seclabel];
     413             :     bool        nulls[Natts_pg_seclabel];
     414             :     bool        replaces[Natts_pg_seclabel];
     415             : 
     416             :     /* Shared objects have their own security label catalog. */
     417          28 :     if (IsSharedRelation(object->classId))
     418             :     {
     419           6 :         SetSharedSecurityLabel(object, provider, label);
     420           6 :         return;
     421             :     }
     422             : 
     423             :     /* Prepare to form or update a tuple, if necessary. */
     424          22 :     memset(nulls, false, sizeof(nulls));
     425          22 :     memset(replaces, false, sizeof(replaces));
     426          22 :     values[Anum_pg_seclabel_objoid - 1] = ObjectIdGetDatum(object->objectId);
     427          22 :     values[Anum_pg_seclabel_classoid - 1] = ObjectIdGetDatum(object->classId);
     428          22 :     values[Anum_pg_seclabel_objsubid - 1] = Int32GetDatum(object->objectSubId);
     429          22 :     values[Anum_pg_seclabel_provider - 1] = CStringGetTextDatum(provider);
     430          22 :     if (label != NULL)
     431          22 :         values[Anum_pg_seclabel_label - 1] = CStringGetTextDatum(label);
     432             : 
     433             :     /* Use the index to search for a matching old tuple */
     434          22 :     ScanKeyInit(&keys[0],
     435             :                 Anum_pg_seclabel_objoid,
     436             :                 BTEqualStrategyNumber, F_OIDEQ,
     437             :                 ObjectIdGetDatum(object->objectId));
     438          22 :     ScanKeyInit(&keys[1],
     439             :                 Anum_pg_seclabel_classoid,
     440             :                 BTEqualStrategyNumber, F_OIDEQ,
     441             :                 ObjectIdGetDatum(object->classId));
     442          22 :     ScanKeyInit(&keys[2],
     443             :                 Anum_pg_seclabel_objsubid,
     444             :                 BTEqualStrategyNumber, F_INT4EQ,
     445             :                 Int32GetDatum(object->objectSubId));
     446          22 :     ScanKeyInit(&keys[3],
     447             :                 Anum_pg_seclabel_provider,
     448             :                 BTEqualStrategyNumber, F_TEXTEQ,
     449          22 :                 CStringGetTextDatum(provider));
     450             : 
     451          22 :     pg_seclabel = table_open(SecLabelRelationId, RowExclusiveLock);
     452             : 
     453          22 :     scan = systable_beginscan(pg_seclabel, SecLabelObjectIndexId, true,
     454             :                               NULL, 4, keys);
     455             : 
     456          22 :     oldtup = systable_getnext(scan);
     457          22 :     if (HeapTupleIsValid(oldtup))
     458             :     {
     459           6 :         if (label == NULL)
     460           0 :             CatalogTupleDelete(pg_seclabel, &oldtup->t_self);
     461             :         else
     462             :         {
     463           6 :             replaces[Anum_pg_seclabel_label - 1] = true;
     464           6 :             newtup = heap_modify_tuple(oldtup, RelationGetDescr(pg_seclabel),
     465             :                                        values, nulls, replaces);
     466           6 :             CatalogTupleUpdate(pg_seclabel, &oldtup->t_self, newtup);
     467             :         }
     468             :     }
     469          22 :     systable_endscan(scan);
     470             : 
     471             :     /* If we didn't find an old tuple, insert a new one */
     472          22 :     if (newtup == NULL && label != NULL)
     473             :     {
     474          16 :         newtup = heap_form_tuple(RelationGetDescr(pg_seclabel),
     475             :                                  values, nulls);
     476          16 :         CatalogTupleInsert(pg_seclabel, newtup);
     477             :     }
     478             : 
     479             :     /* Update indexes, if necessary */
     480          22 :     if (newtup != NULL)
     481          22 :         heap_freetuple(newtup);
     482             : 
     483          22 :     table_close(pg_seclabel, RowExclusiveLock);
     484             : }
     485             : 
     486             : /*
     487             :  * DeleteSharedSecurityLabel is a helper function of DeleteSecurityLabel
     488             :  * to handle shared database objects.
     489             :  */
     490             : void
     491        1464 : DeleteSharedSecurityLabel(Oid objectId, Oid classId)
     492             : {
     493             :     Relation    pg_shseclabel;
     494             :     ScanKeyData skey[2];
     495             :     SysScanDesc scan;
     496             :     HeapTuple   oldtup;
     497             : 
     498        1464 :     ScanKeyInit(&skey[0],
     499             :                 Anum_pg_shseclabel_objoid,
     500             :                 BTEqualStrategyNumber, F_OIDEQ,
     501             :                 ObjectIdGetDatum(objectId));
     502        1464 :     ScanKeyInit(&skey[1],
     503             :                 Anum_pg_shseclabel_classoid,
     504             :                 BTEqualStrategyNumber, F_OIDEQ,
     505             :                 ObjectIdGetDatum(classId));
     506             : 
     507        1464 :     pg_shseclabel = table_open(SharedSecLabelRelationId, RowExclusiveLock);
     508             : 
     509        1464 :     scan = systable_beginscan(pg_shseclabel, SharedSecLabelObjectIndexId, true,
     510             :                               NULL, 2, skey);
     511        1468 :     while (HeapTupleIsValid(oldtup = systable_getnext(scan)))
     512           4 :         CatalogTupleDelete(pg_shseclabel, &oldtup->t_self);
     513        1464 :     systable_endscan(scan);
     514             : 
     515        1464 :     table_close(pg_shseclabel, RowExclusiveLock);
     516        1464 : }
     517             : 
     518             : /*
     519             :  * DeleteSecurityLabel removes all security labels for an object (and any
     520             :  * sub-objects, if applicable).
     521             :  */
     522             : void
     523      200600 : DeleteSecurityLabel(const ObjectAddress *object)
     524             : {
     525             :     Relation    pg_seclabel;
     526             :     ScanKeyData skey[3];
     527             :     SysScanDesc scan;
     528             :     HeapTuple   oldtup;
     529             :     int         nkeys;
     530             : 
     531             :     /* Shared objects have their own security label catalog. */
     532      200600 :     if (IsSharedRelation(object->classId))
     533             :     {
     534             :         Assert(object->objectSubId == 0);
     535           6 :         DeleteSharedSecurityLabel(object->objectId, object->classId);
     536           6 :         return;
     537             :     }
     538             : 
     539      200594 :     ScanKeyInit(&skey[0],
     540             :                 Anum_pg_seclabel_objoid,
     541             :                 BTEqualStrategyNumber, F_OIDEQ,
     542             :                 ObjectIdGetDatum(object->objectId));
     543      200594 :     ScanKeyInit(&skey[1],
     544             :                 Anum_pg_seclabel_classoid,
     545             :                 BTEqualStrategyNumber, F_OIDEQ,
     546             :                 ObjectIdGetDatum(object->classId));
     547      200594 :     if (object->objectSubId != 0)
     548             :     {
     549        1996 :         ScanKeyInit(&skey[2],
     550             :                     Anum_pg_seclabel_objsubid,
     551             :                     BTEqualStrategyNumber, F_INT4EQ,
     552             :                     Int32GetDatum(object->objectSubId));
     553        1996 :         nkeys = 3;
     554             :     }
     555             :     else
     556      198598 :         nkeys = 2;
     557             : 
     558      200594 :     pg_seclabel = table_open(SecLabelRelationId, RowExclusiveLock);
     559             : 
     560      200594 :     scan = systable_beginscan(pg_seclabel, SecLabelObjectIndexId, true,
     561             :                               NULL, nkeys, skey);
     562      200604 :     while (HeapTupleIsValid(oldtup = systable_getnext(scan)))
     563          10 :         CatalogTupleDelete(pg_seclabel, &oldtup->t_self);
     564      200594 :     systable_endscan(scan);
     565             : 
     566      200594 :     table_close(pg_seclabel, RowExclusiveLock);
     567             : }
     568             : 
     569             : void
     570           2 : register_label_provider(const char *provider_name, check_object_relabel_type hook)
     571             : {
     572             :     LabelProvider *provider;
     573             :     MemoryContext oldcxt;
     574             : 
     575           2 :     oldcxt = MemoryContextSwitchTo(TopMemoryContext);
     576           2 :     provider = palloc(sizeof(LabelProvider));
     577           2 :     provider->provider_name = pstrdup(provider_name);
     578           2 :     provider->hook = hook;
     579           2 :     label_provider_list = lappend(label_provider_list, provider);
     580           2 :     MemoryContextSwitchTo(oldcxt);
     581           2 : }

Generated by: LCOV version 1.14