/*-------------------------------------------------------------------------
*
- * oid.c--
- * Functions for the built-in type Oid.
+ * oid.c
+ * Functions for the built-in type Oid ... also oidvector.
*
- * Copyright (c) 1994, Regents of the University of California
+ * Portions Copyright (c) 1996-2014, PostgreSQL Global Development Group
+ * Portions Copyright (c) 1994, Regents of the University of California
*
*
* IDENTIFICATION
- * $Header: /cvsroot/pgsql/src/backend/utils/adt/oid.c,v 1.21 1998/10/29 18:07:03 momjian Exp $
+ * src/backend/utils/adt/oid.c
*
*-------------------------------------------------------------------------
*/
+#include "postgres.h"
-#include <stdio.h>
-#include <string.h>
+#include <ctype.h>
+#include <limits.h>
+
+#include "catalog/pg_type.h"
+#include "libpq/pqformat.h"
+#include "utils/array.h"
+#include "utils/builtins.h"
+
+
+#define OidVectorSize(n) (offsetof(oidvector, values) + (n) * sizeof(Oid))
-#include "postgres.h"
-#include "utils/builtins.h" /* where function declarations go */
/*****************************************************************************
* USER I/O ROUTINES *
*****************************************************************************/
+static Oid
+oidin_subr(const char *s, char **endloc)
+{
+ unsigned long cvt;
+ char *endptr;
+ Oid result;
+
+ if (*s == '\0')
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
+ errmsg("invalid input syntax for type oid: \"%s\"",
+ s)));
+
+ errno = 0;
+ cvt = strtoul(s, &endptr, 10);
+
+ /*
+ * strtoul() normally only sets ERANGE. On some systems it also may set
+ * EINVAL, which simply means it couldn't parse the input string. This is
+ * handled by the second "if" consistent across platforms.
+ */
+ if (errno && errno != ERANGE && errno != EINVAL)
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
+ errmsg("invalid input syntax for type oid: \"%s\"",
+ s)));
+
+ if (endptr == s && *s != '\0')
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
+ errmsg("invalid input syntax for type oid: \"%s\"",
+ s)));
+
+ if (errno == ERANGE)
+ ereport(ERROR,
+ (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
+ errmsg("value \"%s\" is out of range for type oid", s)));
+
+ if (endloc)
+ {
+ /* caller wants to deal with rest of string */
+ *endloc = endptr;
+ }
+ else
+ {
+ /* allow only whitespace after number */
+ while (*endptr && isspace((unsigned char) *endptr))
+ endptr++;
+ if (*endptr)
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
+ errmsg("invalid input syntax for type oid: \"%s\"",
+ s)));
+ }
+
+ result = (Oid) cvt;
+
+ /*
+ * Cope with possibility that unsigned long is wider than Oid, in which
+ * case strtoul will not raise an error for some values that are out of
+ * the range of Oid.
+ *
+ * For backwards compatibility, we want to accept inputs that are given
+ * with a minus sign, so allow the input value if it matches after either
+ * signed or unsigned extension to long.
+ *
+ * To ensure consistent results on 32-bit and 64-bit platforms, make sure
+ * the error message is the same as if strtoul() had returned ERANGE.
+ */
+#if OID_MAX != ULONG_MAX
+ if (cvt != (unsigned long) result &&
+ cvt != (unsigned long) ((int) result))
+ ereport(ERROR,
+ (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
+ errmsg("value \"%s\" is out of range for type oid", s)));
+#endif
+
+ return result;
+}
+
+Datum
+oidin(PG_FUNCTION_ARGS)
+{
+ char *s = PG_GETARG_CSTRING(0);
+ Oid result;
+
+ result = oidin_subr(s, NULL);
+ PG_RETURN_OID(result);
+}
+
+Datum
+oidout(PG_FUNCTION_ARGS)
+{
+ Oid o = PG_GETARG_OID(0);
+ char *result = (char *) palloc(12);
+
+ snprintf(result, 12, "%u", o);
+ PG_RETURN_CSTRING(result);
+}
+
+/*
+ * oidrecv - converts external binary format to oid
+ */
+Datum
+oidrecv(PG_FUNCTION_ARGS)
+{
+ StringInfo buf = (StringInfo) PG_GETARG_POINTER(0);
+
+ PG_RETURN_OID((Oid) pq_getmsgint(buf, sizeof(Oid)));
+}
+
/*
- * oid8in - converts "num num ..." to internal form
+ * oidsend - converts oid to binary format
+ */
+Datum
+oidsend(PG_FUNCTION_ARGS)
+{
+ Oid arg1 = PG_GETARG_OID(0);
+ StringInfoData buf;
+
+ pq_begintypsend(&buf);
+ pq_sendint(&buf, arg1, sizeof(Oid));
+ PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
+}
+
+/*
+ * construct oidvector given a raw array of Oids
*
- * Note:
- * Fills any nonexistent digits with NULL oids.
+ * If oids is NULL then caller must fill values[] afterward
*/
-Oid *
-oid8in(char *oidString)
-{
- Oid *result;
- int nums;
-
- if (oidString == NULL)
- return NULL;
-
- result = (Oid *) palloc(sizeof(Oid[8]));
- if ((nums = sscanf(oidString, "%d%d%d%d%d%d%d%d",
- &result[0],
- &result[1],
- &result[2],
- &result[3],
- &result[4],
- &result[5],
- &result[6],
- &result[7])) != 8)
- {
- do
- result[nums++] = 0;
- while (nums < 8);
- }
+oidvector *
+buildoidvector(const Oid *oids, int n)
+{
+ oidvector *result;
+
+ result = (oidvector *) palloc0(OidVectorSize(n));
+
+ if (n > 0 && oids)
+ memcpy(result->values, oids, n * sizeof(Oid));
+
+ /*
+ * Attach standard array header. For historical reasons, we set the index
+ * lower bound to 0 not 1.
+ */
+ SET_VARSIZE(result, OidVectorSize(n));
+ result->ndim = 1;
+ result->dataoffset = 0; /* never any nulls */
+ result->elemtype = OIDOID;
+ result->dim1 = n;
+ result->lbound1 = 0;
+
return result;
}
/*
- * oid8out - converts internal form to "num num ..."
+ * oidvectorin - converts "num num ..." to internal form
*/
-char *
-oid8out(Oid *oidArray)
+Datum
+oidvectorin(PG_FUNCTION_ARGS)
{
- int num;
- Oid *sp;
- char *rp;
- char *result;
+ char *oidString = PG_GETARG_CSTRING(0);
+ oidvector *result;
+ int n;
+
+ result = (oidvector *) palloc0(OidVectorSize(FUNC_MAX_ARGS));
- if (oidArray == NULL)
+ for (n = 0; n < FUNC_MAX_ARGS; n++)
{
- result = (char *) palloc(2);
- result[0] = '-';
- result[1] = '\0';
- return result;
+ while (*oidString && isspace((unsigned char) *oidString))
+ oidString++;
+ if (*oidString == '\0')
+ break;
+ result->values[n] = oidin_subr(oidString, &oidString);
}
+ while (*oidString && isspace((unsigned char) *oidString))
+ oidString++;
+ if (*oidString)
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
+ errmsg("oidvector has too many elements")));
+
+ SET_VARSIZE(result, OidVectorSize(n));
+ result->ndim = 1;
+ result->dataoffset = 0; /* never any nulls */
+ result->elemtype = OIDOID;
+ result->dim1 = n;
+ result->lbound1 = 0;
+
+ PG_RETURN_POINTER(result);
+}
+
+/*
+ * oidvectorout - converts internal form to "num num ..."
+ */
+Datum
+oidvectorout(PG_FUNCTION_ARGS)
+{
+ oidvector *oidArray = (oidvector *) PG_GETARG_POINTER(0);
+ int num,
+ nnums = oidArray->dim1;
+ char *rp;
+ char *result;
/* assumes sign, 10 digits, ' ' */
- rp = result = (char *) palloc(8 * 12);
- sp = oidArray;
- for (num = 8; num != 0; num--)
+ rp = result = (char *) palloc(nnums * 12 + 1);
+ for (num = 0; num < nnums; num++)
{
- ltoa(*sp++, rp);
+ if (num != 0)
+ *rp++ = ' ';
+ sprintf(rp, "%u", oidArray->values[num]);
while (*++rp != '\0')
;
- *rp++ = ' ';
}
- *--rp = '\0';
- return result;
+ *rp = '\0';
+ PG_RETURN_CSTRING(result);
}
-Oid
-oidin(char *s)
+/*
+ * oidvectorrecv - converts external binary format to oidvector
+ */
+Datum
+oidvectorrecv(PG_FUNCTION_ARGS)
{
- return int4in(s);
+ StringInfo buf = (StringInfo) PG_GETARG_POINTER(0);
+ FunctionCallInfoData locfcinfo;
+ oidvector *result;
+
+ /*
+ * Normally one would call array_recv() using DirectFunctionCall3, but
+ * that does not work since array_recv wants to cache some data using
+ * fcinfo->flinfo->fn_extra. So we need to pass it our own flinfo
+ * parameter.
+ */
+ InitFunctionCallInfoData(locfcinfo, fcinfo->flinfo, 3,
+ InvalidOid, NULL, NULL);
+
+ locfcinfo.arg[0] = PointerGetDatum(buf);
+ locfcinfo.arg[1] = ObjectIdGetDatum(OIDOID);
+ locfcinfo.arg[2] = Int32GetDatum(-1);
+ locfcinfo.argnull[0] = false;
+ locfcinfo.argnull[1] = false;
+ locfcinfo.argnull[2] = false;
+
+ result = (oidvector *) DatumGetPointer(array_recv(&locfcinfo));
+
+ Assert(!locfcinfo.isnull);
+
+ /* sanity checks: oidvector must be 1-D, 0-based, no nulls */
+ if (ARR_NDIM(result) != 1 ||
+ ARR_HASNULL(result) ||
+ ARR_ELEMTYPE(result) != OIDOID ||
+ ARR_LBOUND(result)[0] != 0)
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_BINARY_REPRESENTATION),
+ errmsg("invalid oidvector data")));
+
+ /* check length for consistency with oidvectorin() */
+ if (ARR_DIMS(result)[0] > FUNC_MAX_ARGS)
+ ereport(ERROR,
+ (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
+ errmsg("oidvector has too many elements")));
+
+ PG_RETURN_POINTER(result);
}
-char *
-oidout(Oid o)
+/*
+ * oidvectorsend - converts oidvector to binary format
+ */
+Datum
+oidvectorsend(PG_FUNCTION_ARGS)
{
- return int4out(o);
+ return array_send(fcinfo);
}
-/*****************************************************************************
- * PUBLIC ROUTINES *
- *****************************************************************************/
-
/*
- * If you change this function, change heap_keytest()
- * because we have hardcoded this in there as an optimization
+ * oidparse - get OID from IConst/FConst node
*/
-bool
-oideq(Oid arg1, Oid arg2)
+Oid
+oidparse(Node *node)
{
- return arg1 == arg2;
+ switch (nodeTag(node))
+ {
+ case T_Integer:
+ return intVal(node);
+ case T_Float:
+
+ /*
+ * Values too large for int4 will be represented as Float
+ * constants by the lexer. Accept these if they are valid OID
+ * strings.
+ */
+ return oidin_subr(strVal(node), NULL);
+ default:
+ elog(ERROR, "unrecognized node type: %d", (int) nodeTag(node));
+ }
+ return InvalidOid; /* keep compiler quiet */
}
-bool
-oidne(Oid arg1, Oid arg2)
-{
- return arg1 != arg2;
-}
-bool
-oid8eq(Oid *arg1, Oid *arg2)
+/*****************************************************************************
+ * PUBLIC ROUTINES *
+ *****************************************************************************/
+
+Datum
+oideq(PG_FUNCTION_ARGS)
{
- return (bool) (memcmp(arg1, arg2, 8 * sizeof(Oid)) == 0);
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
+
+ PG_RETURN_BOOL(arg1 == arg2);
}
-bool
-oid8ne(Oid *arg1, Oid *arg2)
+Datum
+oidne(PG_FUNCTION_ARGS)
{
- return (bool) (memcmp(arg1, arg2, 8 * sizeof(Oid)) != 0);
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
+
+ PG_RETURN_BOOL(arg1 != arg2);
}
-bool
-oid8lt(Oid *arg1, Oid *arg2)
+Datum
+oidlt(PG_FUNCTION_ARGS)
{
- int i;
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
- for (i = 0; i < 8; i++)
- if (!int4eq(arg1[i], arg2[i]))
- return int4lt(arg1[i], arg2[i]);
- return false;
+ PG_RETURN_BOOL(arg1 < arg2);
}
-bool
-oid8le(Oid *arg1, Oid *arg2)
+Datum
+oidle(PG_FUNCTION_ARGS)
{
- int i;
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
- for (i = 0; i < 8; i++)
- if (!int4eq(arg1[i], arg2[i]))
- return int4le(arg1[i], arg2[i]);
- return true;
+ PG_RETURN_BOOL(arg1 <= arg2);
}
-bool
-oid8ge(Oid *arg1, Oid *arg2)
+Datum
+oidge(PG_FUNCTION_ARGS)
{
- int i;
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
- for (i = 0; i < 8; i++)
- if (!int4eq(arg1[i], arg2[i]))
- return int4ge(arg1[i], arg2[i]);
- return true;
+ PG_RETURN_BOOL(arg1 >= arg2);
}
-bool
-oid8gt(Oid *arg1, Oid *arg2)
+Datum
+oidgt(PG_FUNCTION_ARGS)
{
- int i;
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
- for (i = 0; i < 8; i++)
- if (!int4eq(arg1[i], arg2[i]))
- return int4gt(arg1[i], arg2[i]);
- return false;
+ PG_RETURN_BOOL(arg1 > arg2);
}
-bool
-oideqint4(Oid arg1, int32 arg2)
+Datum
+oidlarger(PG_FUNCTION_ARGS)
{
-/* oid is unsigned, but int4 is signed */
- return arg2 >= 0 && arg1 == arg2;
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
+
+ PG_RETURN_OID((arg1 > arg2) ? arg1 : arg2);
}
-bool
-int4eqoid(int32 arg1, Oid arg2)
+Datum
+oidsmaller(PG_FUNCTION_ARGS)
{
-/* oid is unsigned, but int4 is signed */
- return arg1 >= 0 && arg1 == arg2;
+ Oid arg1 = PG_GETARG_OID(0);
+ Oid arg2 = PG_GETARG_OID(1);
+
+ PG_RETURN_OID((arg1 < arg2) ? arg1 : arg2);
}
-text *
-oid_text(Oid oid)
+Datum
+oidvectoreq(PG_FUNCTION_ARGS)
{
- text *result;
+ int32 cmp = DatumGetInt32(btoidvectorcmp(fcinfo));
- int len;
- char *str;
+ PG_RETURN_BOOL(cmp == 0);
+}
- str = oidout(oid);
- len = (strlen(str) + VARHDRSZ);
+Datum
+oidvectorne(PG_FUNCTION_ARGS)
+{
+ int32 cmp = DatumGetInt32(btoidvectorcmp(fcinfo));
- result = palloc(len);
+ PG_RETURN_BOOL(cmp != 0);
+}
- VARSIZE(result) = len;
- memmove(VARDATA(result), str, (len - VARHDRSZ));
- pfree(str);
+Datum
+oidvectorlt(PG_FUNCTION_ARGS)
+{
+ int32 cmp = DatumGetInt32(btoidvectorcmp(fcinfo));
- return result;
-} /* oid_text() */
+ PG_RETURN_BOOL(cmp < 0);
+}
-Oid
-text_oid(text *string)
+Datum
+oidvectorle(PG_FUNCTION_ARGS)
{
- Oid result;
+ int32 cmp = DatumGetInt32(btoidvectorcmp(fcinfo));
- int len;
- char *str;
+ PG_RETURN_BOOL(cmp <= 0);
+}
- len = (VARSIZE(string) - VARHDRSZ);
+Datum
+oidvectorge(PG_FUNCTION_ARGS)
+{
+ int32 cmp = DatumGetInt32(btoidvectorcmp(fcinfo));
- str = palloc(len + 1);
- memmove(str, VARDATA(string), len);
- *(str + len) = '\0';
+ PG_RETURN_BOOL(cmp >= 0);
+}
- result = oidin(str);
- pfree(str);
+Datum
+oidvectorgt(PG_FUNCTION_ARGS)
+{
+ int32 cmp = DatumGetInt32(btoidvectorcmp(fcinfo));
- return result;
-} /* oid_text() */
+ PG_RETURN_BOOL(cmp > 0);
+}