Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * nbtvalidate.c
4 : * Opclass validator for btree.
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : * Portions Copyright (c) 1994, Regents of the University of California
8 : *
9 : * IDENTIFICATION
10 : * src/backend/access/nbtree/nbtvalidate.c
11 : *
12 : *-------------------------------------------------------------------------
13 : */
14 : #include "postgres.h"
15 :
16 : #include "access/amvalidate.h"
17 : #include "access/htup_details.h"
18 : #include "access/nbtree.h"
19 : #include "access/xact.h"
20 : #include "catalog/pg_am.h"
21 : #include "catalog/pg_amop.h"
22 : #include "catalog/pg_amproc.h"
23 : #include "catalog/pg_opclass.h"
24 : #include "catalog/pg_opfamily.h"
25 : #include "catalog/pg_type.h"
26 : #include "utils/builtins.h"
27 : #include "utils/lsyscache.h"
28 : #include "utils/regproc.h"
29 : #include "utils/syscache.h"
30 :
31 :
32 : /*
33 : * Validator for a btree opclass.
34 : *
35 : * Some of the checks done here cover the whole opfamily, and therefore are
36 : * redundant when checking each opclass in a family. But they don't run long
37 : * enough to be much of a problem, so we accept the duplication rather than
38 : * complicate the amvalidate API.
39 : */
40 : bool
41 292 : btvalidate(Oid opclassoid)
42 : {
43 292 : bool result = true;
44 : HeapTuple classtup;
45 : Form_pg_opclass classform;
46 : Oid opfamilyoid;
47 : Oid opcintype;
48 : char *opclassname;
49 : HeapTuple familytup;
50 : Form_pg_opfamily familyform;
51 : char *opfamilyname;
52 : CatCList *proclist,
53 : *oprlist;
54 : List *grouplist;
55 : OpFamilyOpFuncGroup *opclassgroup;
56 : List *familytypes;
57 : int usefulgroups;
58 : int i;
59 : ListCell *lc;
60 :
61 : /* Fetch opclass information */
62 292 : classtup = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclassoid));
63 292 : if (!HeapTupleIsValid(classtup))
64 0 : elog(ERROR, "cache lookup failed for operator class %u", opclassoid);
65 292 : classform = (Form_pg_opclass) GETSTRUCT(classtup);
66 :
67 292 : opfamilyoid = classform->opcfamily;
68 292 : opcintype = classform->opcintype;
69 292 : opclassname = NameStr(classform->opcname);
70 :
71 : /* Fetch opfamily information */
72 292 : familytup = SearchSysCache1(OPFAMILYOID, ObjectIdGetDatum(opfamilyoid));
73 292 : if (!HeapTupleIsValid(familytup))
74 0 : elog(ERROR, "cache lookup failed for operator family %u", opfamilyoid);
75 292 : familyform = (Form_pg_opfamily) GETSTRUCT(familytup);
76 :
77 292 : opfamilyname = NameStr(familyform->opfname);
78 :
79 : /* Fetch all operators and support functions of the opfamily */
80 292 : oprlist = SearchSysCacheList1(AMOPSTRATEGY, ObjectIdGetDatum(opfamilyoid));
81 292 : proclist = SearchSysCacheList1(AMPROCNUM, ObjectIdGetDatum(opfamilyoid));
82 :
83 : /* Check individual support functions */
84 2138 : for (i = 0; i < proclist->n_members; i++)
85 : {
86 1846 : HeapTuple proctup = &proclist->members[i]->tuple;
87 1846 : Form_pg_amproc procform = (Form_pg_amproc) GETSTRUCT(proctup);
88 : bool ok;
89 :
90 : /* Check procedure numbers and function signatures */
91 1846 : switch (procform->amprocnum)
92 : {
93 1102 : case BTORDER_PROC:
94 1102 : ok = check_amproc_signature(procform->amproc, INT4OID, true,
95 : 2, 2, procform->amproclefttype,
96 : procform->amprocrighttype);
97 1102 : break;
98 234 : case BTSORTSUPPORT_PROC:
99 234 : ok = check_amproc_signature(procform->amproc, VOIDOID, true,
100 : 1, 1, INTERNALOID);
101 234 : break;
102 228 : case BTINRANGE_PROC:
103 228 : ok = check_amproc_signature(procform->amproc, BOOLOID, true,
104 : 5, 5,
105 : procform->amproclefttype,
106 : procform->amproclefttype,
107 : procform->amprocrighttype,
108 : BOOLOID, BOOLOID);
109 228 : break;
110 282 : case BTEQUALIMAGE_PROC:
111 282 : ok = check_amproc_signature(procform->amproc, BOOLOID, true,
112 : 1, 1, OIDOID);
113 282 : break;
114 0 : case BTOPTIONS_PROC:
115 0 : ok = check_amoptsproc_signature(procform->amproc);
116 0 : break;
117 0 : default:
118 0 : ereport(INFO,
119 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
120 : errmsg("operator family \"%s\" of access method %s contains function %s with invalid support number %d",
121 : opfamilyname, "btree",
122 : format_procedure(procform->amproc),
123 : procform->amprocnum)));
124 0 : result = false;
125 0 : continue; /* don't want additional message */
126 : }
127 :
128 1846 : if (!ok)
129 : {
130 0 : ereport(INFO,
131 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
132 : errmsg("operator family \"%s\" of access method %s contains function %s with wrong signature for support number %d",
133 : opfamilyname, "btree",
134 : format_procedure(procform->amproc),
135 : procform->amprocnum)));
136 0 : result = false;
137 : }
138 : }
139 :
140 : /* Check individual operators */
141 5802 : for (i = 0; i < oprlist->n_members; i++)
142 : {
143 5510 : HeapTuple oprtup = &oprlist->members[i]->tuple;
144 5510 : Form_pg_amop oprform = (Form_pg_amop) GETSTRUCT(oprtup);
145 :
146 : /* Check that only allowed strategy numbers exist */
147 5510 : if (oprform->amopstrategy < 1 ||
148 5510 : oprform->amopstrategy > BTMaxStrategyNumber)
149 : {
150 0 : ereport(INFO,
151 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
152 : errmsg("operator family \"%s\" of access method %s contains operator %s with invalid strategy number %d",
153 : opfamilyname, "btree",
154 : format_operator(oprform->amopopr),
155 : oprform->amopstrategy)));
156 0 : result = false;
157 : }
158 :
159 : /* btree doesn't support ORDER BY operators */
160 5510 : if (oprform->amoppurpose != AMOP_SEARCH ||
161 5510 : OidIsValid(oprform->amopsortfamily))
162 : {
163 0 : ereport(INFO,
164 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
165 : errmsg("operator family \"%s\" of access method %s contains invalid ORDER BY specification for operator %s",
166 : opfamilyname, "btree",
167 : format_operator(oprform->amopopr))));
168 0 : result = false;
169 : }
170 :
171 : /* Check operator signature --- same for all btree strategies */
172 5510 : if (!check_amop_signature(oprform->amopopr, BOOLOID,
173 : oprform->amoplefttype,
174 : oprform->amoprighttype))
175 : {
176 0 : ereport(INFO,
177 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
178 : errmsg("operator family \"%s\" of access method %s contains operator %s with wrong signature",
179 : opfamilyname, "btree",
180 : format_operator(oprform->amopopr))));
181 0 : result = false;
182 : }
183 : }
184 :
185 : /* Now check for inconsistent groups of operators/functions */
186 292 : grouplist = identify_opfamily_groups(oprlist, proclist);
187 292 : usefulgroups = 0;
188 292 : opclassgroup = NULL;
189 292 : familytypes = NIL;
190 1460 : foreach(lc, grouplist)
191 : {
192 1168 : OpFamilyOpFuncGroup *thisgroup = (OpFamilyOpFuncGroup *) lfirst(lc);
193 :
194 : /*
195 : * It is possible for an in_range support function to have a RHS type
196 : * that is otherwise irrelevant to the opfamily --- for instance, SQL
197 : * requires the datetime_ops opclass to have range support with an
198 : * interval offset. So, if this group appears to contain only an
199 : * in_range function, ignore it: it doesn't represent a pair of
200 : * supported types.
201 : */
202 1168 : if (thisgroup->operatorset == 0 &&
203 66 : thisgroup->functionset == (1 << BTINRANGE_PROC))
204 66 : continue;
205 :
206 : /* Else count it as a relevant group */
207 1102 : usefulgroups++;
208 :
209 : /* Remember the group exactly matching the test opclass */
210 1102 : if (thisgroup->lefttype == opcintype &&
211 434 : thisgroup->righttype == opcintype)
212 292 : opclassgroup = thisgroup;
213 :
214 : /*
215 : * Identify all distinct data types handled in this opfamily. This
216 : * implementation is O(N^2), but there aren't likely to be enough
217 : * types in the family for it to matter.
218 : */
219 1102 : familytypes = list_append_unique_oid(familytypes, thisgroup->lefttype);
220 1102 : familytypes = list_append_unique_oid(familytypes, thisgroup->righttype);
221 :
222 : /*
223 : * Complain if there seems to be an incomplete set of either operators
224 : * or support functions for this datatype pair. The sortsupport,
225 : * in_range, and equalimage functions are considered optional.
226 : */
227 1102 : if (thisgroup->operatorset !=
228 : ((1 << BTLessStrategyNumber) |
229 : (1 << BTLessEqualStrategyNumber) |
230 : (1 << BTEqualStrategyNumber) |
231 : (1 << BTGreaterEqualStrategyNumber) |
232 : (1 << BTGreaterStrategyNumber)))
233 : {
234 0 : ereport(INFO,
235 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
236 : errmsg("operator family \"%s\" of access method %s is missing operator(s) for types %s and %s",
237 : opfamilyname, "btree",
238 : format_type_be(thisgroup->lefttype),
239 : format_type_be(thisgroup->righttype))));
240 0 : result = false;
241 : }
242 1102 : if ((thisgroup->functionset & (1 << BTORDER_PROC)) == 0)
243 : {
244 0 : ereport(INFO,
245 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
246 : errmsg("operator family \"%s\" of access method %s is missing support function for types %s and %s",
247 : opfamilyname, "btree",
248 : format_type_be(thisgroup->lefttype),
249 : format_type_be(thisgroup->righttype))));
250 0 : result = false;
251 : }
252 : }
253 :
254 : /* Check that the originally-named opclass is supported */
255 : /* (if group is there, we already checked it adequately above) */
256 292 : if (!opclassgroup)
257 : {
258 0 : ereport(INFO,
259 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
260 : errmsg("operator class \"%s\" of access method %s is missing operator(s)",
261 : opclassname, "btree")));
262 0 : result = false;
263 : }
264 :
265 : /*
266 : * Complain if the opfamily doesn't have entries for all possible
267 : * combinations of its supported datatypes. While missing cross-type
268 : * operators are not fatal, they do limit the planner's ability to derive
269 : * additional qual clauses from equivalence classes, so it seems
270 : * reasonable to insist that all built-in btree opfamilies be complete.
271 : */
272 292 : if (usefulgroups != (list_length(familytypes) * list_length(familytypes)))
273 : {
274 16 : ereport(INFO,
275 : (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
276 : errmsg("operator family \"%s\" of access method %s is missing cross-type operator(s)",
277 : opfamilyname, "btree")));
278 16 : result = false;
279 : }
280 :
281 292 : ReleaseCatCacheList(proclist);
282 292 : ReleaseCatCacheList(oprlist);
283 292 : ReleaseSysCache(familytup);
284 292 : ReleaseSysCache(classtup);
285 :
286 292 : return result;
287 : }
288 :
289 : /*
290 : * Prechecking function for adding operators/functions to a btree opfamily.
291 : */
292 : void
293 208 : btadjustmembers(Oid opfamilyoid,
294 : Oid opclassoid,
295 : List *operators,
296 : List *functions)
297 : {
298 : Oid opcintype;
299 : ListCell *lc;
300 :
301 : /*
302 : * Btree operators and comparison support functions are always "loose"
303 : * members of the opfamily if they are cross-type. If they are not
304 : * cross-type, we prefer to tie them to the appropriate opclass ... but if
305 : * the user hasn't created one, we can't do that, and must fall back to
306 : * using the opfamily dependency. (We mustn't force creation of an
307 : * opclass in such a case, as leaving an incomplete opclass laying about
308 : * would be bad. Throwing an error is another undesirable alternative.)
309 : *
310 : * This behavior results in a bit of a dump/reload hazard, in that the
311 : * order of restoring objects could affect what dependencies we end up
312 : * with. pg_dump's existing behavior will preserve the dependency choices
313 : * in most cases, but not if a cross-type operator has been bound tightly
314 : * into an opclass. That's a mistake anyway, so silently "fixing" it
315 : * isn't awful.
316 : *
317 : * Optional support functions are always "loose" family members.
318 : *
319 : * To avoid repeated lookups, we remember the most recently used opclass's
320 : * input type.
321 : */
322 208 : if (OidIsValid(opclassoid))
323 : {
324 : /* During CREATE OPERATOR CLASS, need CCI to see the pg_opclass row */
325 98 : CommandCounterIncrement();
326 98 : opcintype = get_opclass_input_type(opclassoid);
327 : }
328 : else
329 110 : opcintype = InvalidOid;
330 :
331 : /*
332 : * We handle operators and support functions almost identically, so rather
333 : * than duplicate this code block, just join the lists.
334 : */
335 1194 : foreach(lc, list_concat_copy(operators, functions))
336 : {
337 986 : OpFamilyMember *op = (OpFamilyMember *) lfirst(lc);
338 :
339 986 : if (op->is_func && op->number != BTORDER_PROC)
340 : {
341 : /* Optional support proc, so always a soft family dependency */
342 14 : op->ref_is_hard = false;
343 14 : op->ref_is_family = true;
344 14 : op->refobjid = opfamilyoid;
345 : }
346 972 : else if (op->lefttype != op->righttype)
347 : {
348 : /* Cross-type, so always a soft family dependency */
349 418 : op->ref_is_hard = false;
350 418 : op->ref_is_family = true;
351 418 : op->refobjid = opfamilyoid;
352 : }
353 : else
354 : {
355 : /* Not cross-type; is there a suitable opclass? */
356 554 : if (op->lefttype != opcintype)
357 : {
358 : /* Avoid repeating this expensive lookup, even if it fails */
359 40 : opcintype = op->lefttype;
360 40 : opclassoid = opclass_for_family_datatype(BTREE_AM_OID,
361 : opfamilyoid,
362 : opcintype);
363 : }
364 554 : if (OidIsValid(opclassoid))
365 : {
366 : /* Hard dependency on opclass */
367 514 : op->ref_is_hard = true;
368 514 : op->ref_is_family = false;
369 514 : op->refobjid = opclassoid;
370 : }
371 : else
372 : {
373 : /* We're stuck, so make a soft dependency on the opfamily */
374 40 : op->ref_is_hard = false;
375 40 : op->ref_is_family = true;
376 40 : op->refobjid = opfamilyoid;
377 : }
378 : }
379 : }
380 208 : }
|