23#include "utils/fmgrprotos.h"
31#define NS_PER_S INT64CONST(1000000000)
32#define NS_PER_MS INT64CONST(1000000)
33#define NS_PER_US INT64CONST(1000)
34#define US_PER_MS INT64CONST(1000)
51#if defined(__darwin__) || defined(_MSC_VER)
52#define SUBMS_MINIMAL_STEP_BITS 10
54#define SUBMS_MINIMAL_STEP_BITS 12
57#define SUBMS_MINIMAL_STEP_NS ((NS_PER_MS / (1 << SUBMS_MINIMAL_STEP_BITS)) + 1)
92 static const char hex_chars[] =
"0123456789abcdef";
110 if (
i == 4 ||
i == 6 ||
i == 8 ||
i == 10)
113 hi = uuid->
data[
i] >> 4;
114 lo = uuid->
data[
i] & 0x0F;
116 *p++ = hex_chars[hi];
117 *p++ = hex_chars[lo];
147 if (src[0] ==
'\0' || src[1] ==
'\0')
149 memcpy(str_buf, src, 2);
150 if (!isxdigit((
unsigned char) str_buf[0]) ||
151 !isxdigit((
unsigned char) str_buf[1]))
155 uuid->
data[
i] = (
unsigned char) strtoul(str_buf, NULL, 16);
157 if (src[0] ==
'-' && (
i % 2) == 1 &&
i <
UUID_LEN - 1)
175 (
errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
176 errmsg(
"invalid input syntax for type %s: \"%s\"",
333 if (memtupcount < 10000 || uss->input_count < 10000 || !uss->estimating)
344 if (abbr_card > 100000.0)
348 "uuid_abbrev: estimation ends at cardinality %f"
361 if (abbr_card < uss->input_count / 2000.0 + 0.5)
365 "uuid_abbrev: aborting abbreviation at cardinality %f"
366 " below threshold %f after " INT64_FORMAT " values (%d rows)",
375 " values (%d rows)", abbr_card, uss->
input_count, memtupcount);
394 memcpy(&res, authoritative->
data,
sizeof(
Datum));
418 res = DatumBigEndianToNative(res);
432 if (uuid->
data[
i] > 0)
438 uuid->
data[
i] = UCHAR_MAX;
457 if (uuid->
data[
i] < UCHAR_MAX)
515 uuid->
data[6] = (uuid->
data[6] & 0x0f) | (version << 4);
518 uuid->
data[8] = (uuid->
data[8] & 0x3f) | 0x80;
534 (
errcode(ERRCODE_INTERNAL_ERROR),
535 errmsg(
"could not generate random values")));
554 static int64 previous_ns = 0;
576 clock_gettime(CLOCK_REALTIME, &tmp);
577 ns = tmp.tv_sec *
NS_PER_S + tmp.tv_nsec;
608 uint32 increased_clock_precision;
611 uuid->
data[0] = (
unsigned char) (unix_ts_ms >> 40);
612 uuid->
data[1] = (
unsigned char) (unix_ts_ms >> 32);
613 uuid->
data[2] = (
unsigned char) (unix_ts_ms >> 24);
614 uuid->
data[3] = (
unsigned char) (unix_ts_ms >> 16);
615 uuid->
data[4] = (
unsigned char) (unix_ts_ms >> 8);
616 uuid->
data[5] = (
unsigned char) unix_ts_ms;
625 uuid->
data[6] = (
unsigned char) (increased_clock_precision >> 8);
626 uuid->
data[7] = (
unsigned char) (increased_clock_precision);
631 (
errcode(ERRCODE_INTERNAL_ERROR),
632 errmsg(
"could not generate random values")));
634#if SUBMS_MINIMAL_STEP_BITS == 10
707#define GREGORIAN_EPOCH_JDATE INT64CONST(2299161)
723 if ((uuid->
data[8] & 0xc0) != 0x80)
726 version = uuid->
data[6] >> 4;
747 tms = (uuid->
data[5])
777 if ((uuid->
data[8] & 0xc0) != 0x80)
780 version = uuid->
data[6] >> 4;
Datum timestamptz_pl_interval(PG_FUNCTION_ARGS)
#define POSTGRES_EPOCH_JDATE
int errcode(int sqlerrcode)
int errmsg(const char *fmt,...)
#define ereturn(context, dummy_value,...)
#define ereport(elevel,...)
#define PG_RETURN_BYTEA_P(x)
#define DirectFunctionCall2(func, arg1, arg2)
#define PG_GETARG_POINTER(n)
#define PG_RETURN_CSTRING(x)
#define PG_GETARG_CSTRING(n)
#define PG_GETARG_INT64(n)
#define PG_RETURN_INT32(x)
#define PG_RETURN_UINT16(x)
#define PG_RETURN_POINTER(x)
#define PG_RETURN_BOOL(x)
static Datum hash_uint32(uint32 k)
static Datum hash_any_extended(const unsigned char *k, int keylen, uint64 seed)
static Datum hash_any(const unsigned char *k, int keylen)
void initHyperLogLog(hyperLogLogState *cState, uint8 bwidth)
double estimateHyperLogLog(hyperLogLogState *cState)
void addHyperLogLog(hyperLogLogState *cState, uint32 hash)
void pfree(void *pointer)
static MemoryContext MemoryContextSwitchTo(MemoryContext context)
static rewind_source * source
void syntax_error(const char *source, int lineno, const char *line, const char *command, const char *msg, const char *more, int column)
bool pg_strong_random(void *buf, size_t len)
static uint32 DatumGetUInt32(Datum X)
struct SkipSupportData * SkipSupport
struct SortSupportData * SortSupport
StringInfoData * StringInfo
SkipSupportIncDec decrement
SkipSupportIncDec increment
int(* comparator)(Datum x, Datum y, SortSupport ssup)
Datum(* abbrev_converter)(Datum original, SortSupport ssup)
int(* abbrev_full_comparator)(Datum x, Datum y, SortSupport ssup)
bool(* abbrev_abort)(int memtupcount, SortSupport ssup)
unsigned char data[UUID_LEN]
hyperLogLogState abbr_card
int ssup_datum_unsigned_cmp(Datum x, Datum y, SortSupport ssup)
static Datum TimestampTzGetDatum(TimestampTz X)
static Datum IntervalPGetDatum(const Interval *X)
#define PG_GETARG_INTERVAL_P(n)
#define PG_RETURN_TIMESTAMPTZ(x)
static TimestampTz DatumGetTimestampTz(Datum X)
Datum uuid_skipsupport(PG_FUNCTION_ARGS)
static int64 get_real_time_ns_ascending()
Datum uuid_send(PG_FUNCTION_ARGS)
static void string_to_uuid(const char *source, pg_uuid_t *uuid, Node *escontext)
static bool uuid_abbrev_abort(int memtupcount, SortSupport ssup)
#define GREGORIAN_EPOCH_JDATE
Datum uuidv7_interval(PG_FUNCTION_ARGS)
Datum uuid_lt(PG_FUNCTION_ARGS)
Datum uuid_gt(PG_FUNCTION_ARGS)
Datum gen_random_uuid(PG_FUNCTION_ARGS)
Datum uuid_recv(PG_FUNCTION_ARGS)
Datum uuid_cmp(PG_FUNCTION_ARGS)
static void uuid_set_version(pg_uuid_t *uuid, unsigned char version)
Datum uuid_le(PG_FUNCTION_ARGS)
#define SUBMS_MINIMAL_STEP_NS
Datum uuid_hash(PG_FUNCTION_ARGS)
Datum uuid_extract_version(PG_FUNCTION_ARGS)
Datum uuid_hash_extended(PG_FUNCTION_ARGS)
Datum uuid_out(PG_FUNCTION_ARGS)
Datum uuid_ne(PG_FUNCTION_ARGS)
static int uuid_internal_cmp(const pg_uuid_t *arg1, const pg_uuid_t *arg2)
Datum uuid_ge(PG_FUNCTION_ARGS)
static Datum uuid_increment(Relation rel, Datum existing, bool *overflow)
Datum uuid_eq(PG_FUNCTION_ARGS)
static pg_uuid_t * generate_uuidv7(uint64 unix_ts_ms, uint32 sub_ms)
static int uuid_fast_cmp(Datum x, Datum y, SortSupport ssup)
Datum uuid_in(PG_FUNCTION_ARGS)
static Datum uuid_decrement(Relation rel, Datum existing, bool *underflow)
static Datum uuid_abbrev_convert(Datum original, SortSupport ssup)
Datum uuidv7(PG_FUNCTION_ARGS)
Datum uuid_extract_timestamp(PG_FUNCTION_ARGS)
Datum uuid_sortsupport(PG_FUNCTION_ARGS)
static pg_uuid_t * DatumGetUUIDP(Datum X)
#define PG_RETURN_UUID_P(X)
static Datum UUIDPGetDatum(const pg_uuid_t *X)
#define PG_GETARG_UUID_P(X)
int gettimeofday(struct timeval *tp, void *tzp)