1 | /*------------------------------------------------------------------------- |
2 | * |
3 | * gistvalidate.c |
4 | * Opclass validator for GiST. |
5 | * |
6 | * Portions Copyright (c) 1996-2019, PostgreSQL Global Development Group |
7 | * Portions Copyright (c) 1994, Regents of the University of California |
8 | * |
9 | * IDENTIFICATION |
10 | * src/backend/access/gist/gistvalidate.c |
11 | * |
12 | *------------------------------------------------------------------------- |
13 | */ |
14 | #include "postgres.h" |
15 | |
16 | #include "access/amvalidate.h" |
17 | #include "access/gist_private.h" |
18 | #include "access/htup_details.h" |
19 | #include "catalog/pg_amop.h" |
20 | #include "catalog/pg_amproc.h" |
21 | #include "catalog/pg_opclass.h" |
22 | #include "catalog/pg_opfamily.h" |
23 | #include "catalog/pg_type.h" |
24 | #include "utils/builtins.h" |
25 | #include "utils/lsyscache.h" |
26 | #include "utils/regproc.h" |
27 | #include "utils/syscache.h" |
28 | |
29 | |
30 | /* |
31 | * Validator for a GiST opclass. |
32 | */ |
33 | bool |
34 | gistvalidate(Oid opclassoid) |
35 | { |
36 | bool result = true; |
37 | HeapTuple classtup; |
38 | Form_pg_opclass classform; |
39 | Oid opfamilyoid; |
40 | Oid opcintype; |
41 | Oid opckeytype; |
42 | char *opclassname; |
43 | HeapTuple familytup; |
44 | Form_pg_opfamily familyform; |
45 | char *opfamilyname; |
46 | CatCList *proclist, |
47 | *oprlist; |
48 | List *grouplist; |
49 | OpFamilyOpFuncGroup *opclassgroup; |
50 | int i; |
51 | ListCell *lc; |
52 | |
53 | /* Fetch opclass information */ |
54 | classtup = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclassoid)); |
55 | if (!HeapTupleIsValid(classtup)) |
56 | elog(ERROR, "cache lookup failed for operator class %u" , opclassoid); |
57 | classform = (Form_pg_opclass) GETSTRUCT(classtup); |
58 | |
59 | opfamilyoid = classform->opcfamily; |
60 | opcintype = classform->opcintype; |
61 | opckeytype = classform->opckeytype; |
62 | if (!OidIsValid(opckeytype)) |
63 | opckeytype = opcintype; |
64 | opclassname = NameStr(classform->opcname); |
65 | |
66 | /* Fetch opfamily information */ |
67 | familytup = SearchSysCache1(OPFAMILYOID, ObjectIdGetDatum(opfamilyoid)); |
68 | if (!HeapTupleIsValid(familytup)) |
69 | elog(ERROR, "cache lookup failed for operator family %u" , opfamilyoid); |
70 | familyform = (Form_pg_opfamily) GETSTRUCT(familytup); |
71 | |
72 | opfamilyname = NameStr(familyform->opfname); |
73 | |
74 | /* Fetch all operators and support functions of the opfamily */ |
75 | oprlist = SearchSysCacheList1(AMOPSTRATEGY, ObjectIdGetDatum(opfamilyoid)); |
76 | proclist = SearchSysCacheList1(AMPROCNUM, ObjectIdGetDatum(opfamilyoid)); |
77 | |
78 | /* Check individual support functions */ |
79 | for (i = 0; i < proclist->n_members; i++) |
80 | { |
81 | HeapTuple proctup = &proclist->members[i]->tuple; |
82 | Form_pg_amproc procform = (Form_pg_amproc) GETSTRUCT(proctup); |
83 | bool ok; |
84 | |
85 | /* |
86 | * All GiST support functions should be registered with matching |
87 | * left/right types |
88 | */ |
89 | if (procform->amproclefttype != procform->amprocrighttype) |
90 | { |
91 | ereport(INFO, |
92 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
93 | errmsg("operator family \"%s\" of access method %s contains support function %s with different left and right input types" , |
94 | opfamilyname, "gist" , |
95 | format_procedure(procform->amproc)))); |
96 | result = false; |
97 | } |
98 | |
99 | /* |
100 | * We can't check signatures except within the specific opclass, since |
101 | * we need to know the associated opckeytype in many cases. |
102 | */ |
103 | if (procform->amproclefttype != opcintype) |
104 | continue; |
105 | |
106 | /* Check procedure numbers and function signatures */ |
107 | switch (procform->amprocnum) |
108 | { |
109 | case GIST_CONSISTENT_PROC: |
110 | ok = check_amproc_signature(procform->amproc, BOOLOID, false, |
111 | 5, 5, INTERNALOID, opcintype, |
112 | INT2OID, OIDOID, INTERNALOID); |
113 | break; |
114 | case GIST_UNION_PROC: |
115 | ok = check_amproc_signature(procform->amproc, opckeytype, false, |
116 | 2, 2, INTERNALOID, INTERNALOID); |
117 | break; |
118 | case GIST_COMPRESS_PROC: |
119 | case GIST_DECOMPRESS_PROC: |
120 | case GIST_FETCH_PROC: |
121 | ok = check_amproc_signature(procform->amproc, INTERNALOID, true, |
122 | 1, 1, INTERNALOID); |
123 | break; |
124 | case GIST_PENALTY_PROC: |
125 | ok = check_amproc_signature(procform->amproc, INTERNALOID, true, |
126 | 3, 3, INTERNALOID, |
127 | INTERNALOID, INTERNALOID); |
128 | break; |
129 | case GIST_PICKSPLIT_PROC: |
130 | ok = check_amproc_signature(procform->amproc, INTERNALOID, true, |
131 | 2, 2, INTERNALOID, INTERNALOID); |
132 | break; |
133 | case GIST_EQUAL_PROC: |
134 | ok = check_amproc_signature(procform->amproc, INTERNALOID, false, |
135 | 3, 3, opckeytype, opckeytype, |
136 | INTERNALOID); |
137 | break; |
138 | case GIST_DISTANCE_PROC: |
139 | ok = check_amproc_signature(procform->amproc, FLOAT8OID, false, |
140 | 5, 5, INTERNALOID, opcintype, |
141 | INT2OID, OIDOID, INTERNALOID); |
142 | break; |
143 | default: |
144 | ereport(INFO, |
145 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
146 | errmsg("operator family \"%s\" of access method %s contains function %s with invalid support number %d" , |
147 | opfamilyname, "gist" , |
148 | format_procedure(procform->amproc), |
149 | procform->amprocnum))); |
150 | result = false; |
151 | continue; /* don't want additional message */ |
152 | } |
153 | |
154 | if (!ok) |
155 | { |
156 | ereport(INFO, |
157 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
158 | errmsg("operator family \"%s\" of access method %s contains function %s with wrong signature for support number %d" , |
159 | opfamilyname, "gist" , |
160 | format_procedure(procform->amproc), |
161 | procform->amprocnum))); |
162 | result = false; |
163 | } |
164 | } |
165 | |
166 | /* Check individual operators */ |
167 | for (i = 0; i < oprlist->n_members; i++) |
168 | { |
169 | HeapTuple oprtup = &oprlist->members[i]->tuple; |
170 | Form_pg_amop oprform = (Form_pg_amop) GETSTRUCT(oprtup); |
171 | Oid op_rettype; |
172 | |
173 | /* TODO: Check that only allowed strategy numbers exist */ |
174 | if (oprform->amopstrategy < 1) |
175 | { |
176 | ereport(INFO, |
177 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
178 | errmsg("operator family \"%s\" of access method %s contains operator %s with invalid strategy number %d" , |
179 | opfamilyname, "gist" , |
180 | format_operator(oprform->amopopr), |
181 | oprform->amopstrategy))); |
182 | result = false; |
183 | } |
184 | |
185 | /* GiST supports ORDER BY operators */ |
186 | if (oprform->amoppurpose != AMOP_SEARCH) |
187 | { |
188 | /* ... but must have matching distance proc */ |
189 | if (!OidIsValid(get_opfamily_proc(opfamilyoid, |
190 | oprform->amoplefttype, |
191 | oprform->amoplefttype, |
192 | GIST_DISTANCE_PROC))) |
193 | { |
194 | ereport(INFO, |
195 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
196 | errmsg("operator family \"%s\" of access method %s contains unsupported ORDER BY specification for operator %s" , |
197 | opfamilyname, "gist" , |
198 | format_operator(oprform->amopopr)))); |
199 | result = false; |
200 | } |
201 | /* ... and operator result must match the claimed btree opfamily */ |
202 | op_rettype = get_op_rettype(oprform->amopopr); |
203 | if (!opfamily_can_sort_type(oprform->amopsortfamily, op_rettype)) |
204 | { |
205 | ereport(INFO, |
206 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
207 | errmsg("operator family \"%s\" of access method %s contains incorrect ORDER BY opfamily specification for operator %s" , |
208 | opfamilyname, "gist" , |
209 | format_operator(oprform->amopopr)))); |
210 | result = false; |
211 | } |
212 | } |
213 | else |
214 | { |
215 | /* Search operators must always return bool */ |
216 | op_rettype = BOOLOID; |
217 | } |
218 | |
219 | /* Check operator signature */ |
220 | if (!check_amop_signature(oprform->amopopr, op_rettype, |
221 | oprform->amoplefttype, |
222 | oprform->amoprighttype)) |
223 | { |
224 | ereport(INFO, |
225 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
226 | errmsg("operator family \"%s\" of access method %s contains operator %s with wrong signature" , |
227 | opfamilyname, "gist" , |
228 | format_operator(oprform->amopopr)))); |
229 | result = false; |
230 | } |
231 | } |
232 | |
233 | /* Now check for inconsistent groups of operators/functions */ |
234 | grouplist = identify_opfamily_groups(oprlist, proclist); |
235 | opclassgroup = NULL; |
236 | foreach(lc, grouplist) |
237 | { |
238 | OpFamilyOpFuncGroup *thisgroup = (OpFamilyOpFuncGroup *) lfirst(lc); |
239 | |
240 | /* Remember the group exactly matching the test opclass */ |
241 | if (thisgroup->lefttype == opcintype && |
242 | thisgroup->righttype == opcintype) |
243 | opclassgroup = thisgroup; |
244 | |
245 | /* |
246 | * There is not a lot we can do to check the operator sets, since each |
247 | * GiST opclass is more or less a law unto itself, and some contain |
248 | * only operators that are binary-compatible with the opclass datatype |
249 | * (meaning that empty operator sets can be OK). That case also means |
250 | * that we shouldn't insist on nonempty function sets except for the |
251 | * opclass's own group. |
252 | */ |
253 | } |
254 | |
255 | /* Check that the originally-named opclass is complete */ |
256 | for (i = 1; i <= GISTNProcs; i++) |
257 | { |
258 | if (opclassgroup && |
259 | (opclassgroup->functionset & (((uint64) 1) << i)) != 0) |
260 | continue; /* got it */ |
261 | if (i == GIST_DISTANCE_PROC || i == GIST_FETCH_PROC || |
262 | i == GIST_COMPRESS_PROC || i == GIST_DECOMPRESS_PROC) |
263 | continue; /* optional methods */ |
264 | ereport(INFO, |
265 | (errcode(ERRCODE_INVALID_OBJECT_DEFINITION), |
266 | errmsg("operator class \"%s\" of access method %s is missing support function %d" , |
267 | opclassname, "gist" , i))); |
268 | result = false; |
269 | } |
270 | |
271 | ReleaseCatCacheList(proclist); |
272 | ReleaseCatCacheList(oprlist); |
273 | ReleaseSysCache(familytup); |
274 | ReleaseSysCache(classtup); |
275 | |
276 | return result; |
277 | } |
278 | |