mirror of
https://github.com/postgres/postgres.git
synced 2025-04-25 21:42:33 +03:00
Add to_bin() and to_oct().
This commit introduces functions for converting numbers to their equivalent binary and octal representations. Also, the base conversion code for these functions and to_hex() has been moved to a common helper function. Co-authored-by: Eric Radman Reviewed-by: Ian Barwick, Dag Lem, Vignesh C, Tom Lane, Peter Eisentraut, Kirk Wolak, Vik Fearing, John Naylor, Dean Rasheed Discussion: https://postgr.es/m/Y6IyTQQ/TsD5wnsH%40vm3.eradman.com
This commit is contained in:
parent
ccadf73163
commit
260a1f18da
@ -3737,6 +3737,32 @@ repeat('Pg', 4) <returnvalue>PgPgPgPg</returnvalue>
|
|||||||
</para></entry>
|
</para></entry>
|
||||||
</row>
|
</row>
|
||||||
|
|
||||||
|
<row>
|
||||||
|
<entry role="func_table_entry"><para role="func_signature">
|
||||||
|
<indexterm>
|
||||||
|
<primary>to_bin</primary>
|
||||||
|
</indexterm>
|
||||||
|
<function>to_bin</function> ( <type>integer</type> )
|
||||||
|
<returnvalue>text</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para role="func_signature">
|
||||||
|
<function>to_bin</function> ( <type>bigint</type> )
|
||||||
|
<returnvalue>text</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
Converts the number to its equivalent two's complement binary
|
||||||
|
representation.
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
<literal>to_bin(2147483647)</literal>
|
||||||
|
<returnvalue>1111111111111111111111111111111</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
<literal>to_bin(-1234)</literal>
|
||||||
|
<returnvalue>11111111111111111111101100101110</returnvalue>
|
||||||
|
</para></entry>
|
||||||
|
</row>
|
||||||
|
|
||||||
<row>
|
<row>
|
||||||
<entry role="func_table_entry"><para role="func_signature">
|
<entry role="func_table_entry"><para role="func_signature">
|
||||||
<indexterm>
|
<indexterm>
|
||||||
@ -3750,11 +3776,42 @@ repeat('Pg', 4) <returnvalue>PgPgPgPg</returnvalue>
|
|||||||
<returnvalue>text</returnvalue>
|
<returnvalue>text</returnvalue>
|
||||||
</para>
|
</para>
|
||||||
<para>
|
<para>
|
||||||
Converts the number to its equivalent hexadecimal representation.
|
Converts the number to its equivalent two's complement hexadecimal
|
||||||
|
representation.
|
||||||
</para>
|
</para>
|
||||||
<para>
|
<para>
|
||||||
<literal>to_hex(2147483647)</literal>
|
<literal>to_hex(2147483647)</literal>
|
||||||
<returnvalue>7fffffff</returnvalue>
|
<returnvalue>7fffffff</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
<literal>to_hex(-1234)</literal>
|
||||||
|
<returnvalue>fffffb2e</returnvalue>
|
||||||
|
</para></entry>
|
||||||
|
</row>
|
||||||
|
|
||||||
|
<row>
|
||||||
|
<entry role="func_table_entry"><para role="func_signature">
|
||||||
|
<indexterm>
|
||||||
|
<primary>to_oct</primary>
|
||||||
|
</indexterm>
|
||||||
|
<function>to_oct</function> ( <type>integer</type> )
|
||||||
|
<returnvalue>text</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para role="func_signature">
|
||||||
|
<function>to_oct</function> ( <type>bigint</type> )
|
||||||
|
<returnvalue>text</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
Converts the number to its equivalent two's complement octal
|
||||||
|
representation.
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
<literal>to_oct(2147483647)</literal>
|
||||||
|
<returnvalue>17777777777</returnvalue>
|
||||||
|
</para>
|
||||||
|
<para>
|
||||||
|
<literal>to_oct(-1234)</literal>
|
||||||
|
<returnvalue>37777775456</returnvalue>
|
||||||
</para></entry>
|
</para></entry>
|
||||||
</row>
|
</row>
|
||||||
|
|
||||||
|
@ -4919,53 +4919,87 @@ array_to_text_internal(FunctionCallInfo fcinfo, ArrayType *v,
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define HEXBASE 16
|
|
||||||
/*
|
/*
|
||||||
* Convert an int32 to a string containing a base 16 (hex) representation of
|
* Workhorse for to_bin, to_oct, and to_hex. Note that base must be > 1 and <=
|
||||||
|
* 16.
|
||||||
|
*/
|
||||||
|
static inline text *
|
||||||
|
convert_to_base(uint64 value, int base)
|
||||||
|
{
|
||||||
|
const char *digits = "0123456789abcdef";
|
||||||
|
|
||||||
|
/* We size the buffer for to_bin's longest possible return value. */
|
||||||
|
char buf[sizeof(uint64) * BITS_PER_BYTE];
|
||||||
|
char *const end = buf + sizeof(buf);
|
||||||
|
char *ptr = end;
|
||||||
|
|
||||||
|
Assert(base > 1);
|
||||||
|
Assert(base <= 16);
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
*--ptr = digits[value % base];
|
||||||
|
value /= base;
|
||||||
|
} while (ptr > buf && value);
|
||||||
|
|
||||||
|
return cstring_to_text_with_len(ptr, end - ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Convert an integer to a string containing a base-2 (binary) representation
|
||||||
|
* of the number.
|
||||||
|
*/
|
||||||
|
Datum
|
||||||
|
to_bin32(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
uint64 value = (uint32) PG_GETARG_INT32(0);
|
||||||
|
|
||||||
|
PG_RETURN_TEXT_P(convert_to_base(value, 2));
|
||||||
|
}
|
||||||
|
Datum
|
||||||
|
to_bin64(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
uint64 value = (uint64) PG_GETARG_INT64(0);
|
||||||
|
|
||||||
|
PG_RETURN_TEXT_P(convert_to_base(value, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Convert an integer to a string containing a base-8 (oct) representation of
|
||||||
|
* the number.
|
||||||
|
*/
|
||||||
|
Datum
|
||||||
|
to_oct32(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
uint64 value = (uint32) PG_GETARG_INT32(0);
|
||||||
|
|
||||||
|
PG_RETURN_TEXT_P(convert_to_base(value, 8));
|
||||||
|
}
|
||||||
|
Datum
|
||||||
|
to_oct64(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
uint64 value = (uint64) PG_GETARG_INT64(0);
|
||||||
|
|
||||||
|
PG_RETURN_TEXT_P(convert_to_base(value, 8));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Convert an integer to a string containing a base-16 (hex) representation of
|
||||||
* the number.
|
* the number.
|
||||||
*/
|
*/
|
||||||
Datum
|
Datum
|
||||||
to_hex32(PG_FUNCTION_ARGS)
|
to_hex32(PG_FUNCTION_ARGS)
|
||||||
{
|
{
|
||||||
uint32 value = (uint32) PG_GETARG_INT32(0);
|
uint64 value = (uint32) PG_GETARG_INT32(0);
|
||||||
char *ptr;
|
|
||||||
const char *digits = "0123456789abcdef";
|
|
||||||
char buf[32]; /* bigger than needed, but reasonable */
|
|
||||||
|
|
||||||
ptr = buf + sizeof(buf) - 1;
|
PG_RETURN_TEXT_P(convert_to_base(value, 16));
|
||||||
*ptr = '\0';
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
*--ptr = digits[value % HEXBASE];
|
|
||||||
value /= HEXBASE;
|
|
||||||
} while (ptr > buf && value);
|
|
||||||
|
|
||||||
PG_RETURN_TEXT_P(cstring_to_text(ptr));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Convert an int64 to a string containing a base 16 (hex) representation of
|
|
||||||
* the number.
|
|
||||||
*/
|
|
||||||
Datum
|
Datum
|
||||||
to_hex64(PG_FUNCTION_ARGS)
|
to_hex64(PG_FUNCTION_ARGS)
|
||||||
{
|
{
|
||||||
uint64 value = (uint64) PG_GETARG_INT64(0);
|
uint64 value = (uint64) PG_GETARG_INT64(0);
|
||||||
char *ptr;
|
|
||||||
const char *digits = "0123456789abcdef";
|
|
||||||
char buf[32]; /* bigger than needed, but reasonable */
|
|
||||||
|
|
||||||
ptr = buf + sizeof(buf) - 1;
|
PG_RETURN_TEXT_P(convert_to_base(value, 16));
|
||||||
*ptr = '\0';
|
|
||||||
|
|
||||||
do
|
|
||||||
{
|
|
||||||
*--ptr = digits[value % HEXBASE];
|
|
||||||
value /= HEXBASE;
|
|
||||||
} while (ptr > buf && value);
|
|
||||||
|
|
||||||
PG_RETURN_TEXT_P(cstring_to_text(ptr));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -3707,6 +3707,18 @@
|
|||||||
{ oid => '2768', descr => 'split string by pattern',
|
{ oid => '2768', descr => 'split string by pattern',
|
||||||
proname => 'regexp_split_to_array', prorettype => '_text',
|
proname => 'regexp_split_to_array', prorettype => '_text',
|
||||||
proargtypes => 'text text text', prosrc => 'regexp_split_to_array' },
|
proargtypes => 'text text text', prosrc => 'regexp_split_to_array' },
|
||||||
|
{ oid => '9030', descr => 'convert int4 number to binary',
|
||||||
|
proname => 'to_bin', prorettype => 'text', proargtypes => 'int4',
|
||||||
|
prosrc => 'to_bin32' },
|
||||||
|
{ oid => '9031', descr => 'convert int8 number to binary',
|
||||||
|
proname => 'to_bin', prorettype => 'text', proargtypes => 'int8',
|
||||||
|
prosrc => 'to_bin64' },
|
||||||
|
{ oid => '9032', descr => 'convert int4 number to oct',
|
||||||
|
proname => 'to_oct', prorettype => 'text', proargtypes => 'int4',
|
||||||
|
prosrc => 'to_oct32' },
|
||||||
|
{ oid => '9033', descr => 'convert int8 number to oct',
|
||||||
|
proname => 'to_oct', prorettype => 'text', proargtypes => 'int8',
|
||||||
|
prosrc => 'to_oct64' },
|
||||||
{ oid => '2089', descr => 'convert int4 number to hex',
|
{ oid => '2089', descr => 'convert int4 number to hex',
|
||||||
proname => 'to_hex', prorettype => 'text', proargtypes => 'int4',
|
proname => 'to_hex', prorettype => 'text', proargtypes => 'int4',
|
||||||
prosrc => 'to_hex32' },
|
prosrc => 'to_hex32' },
|
||||||
|
@ -2129,8 +2129,68 @@ select split_part('@joeuser@mydatabase@','@',-2) AS "mydatabase";
|
|||||||
(1 row)
|
(1 row)
|
||||||
|
|
||||||
--
|
--
|
||||||
-- test to_hex
|
-- test to_bin, to_oct, and to_hex
|
||||||
--
|
--
|
||||||
|
select to_bin(-1234) AS "11111111111111111111101100101110";
|
||||||
|
11111111111111111111101100101110
|
||||||
|
----------------------------------
|
||||||
|
11111111111111111111101100101110
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_bin(-1234::bigint);
|
||||||
|
to_bin
|
||||||
|
------------------------------------------------------------------
|
||||||
|
1111111111111111111111111111111111111111111111111111101100101110
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_bin(256*256*256 - 1) AS "111111111111111111111111";
|
||||||
|
111111111111111111111111
|
||||||
|
--------------------------
|
||||||
|
111111111111111111111111
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_bin(256::bigint*256::bigint*256::bigint*256::bigint - 1) AS "11111111111111111111111111111111";
|
||||||
|
11111111111111111111111111111111
|
||||||
|
----------------------------------
|
||||||
|
11111111111111111111111111111111
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_oct(-1234) AS "37777775456";
|
||||||
|
37777775456
|
||||||
|
-------------
|
||||||
|
37777775456
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_oct(-1234::bigint) AS "1777777777777777775456";
|
||||||
|
1777777777777777775456
|
||||||
|
------------------------
|
||||||
|
1777777777777777775456
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_oct(256*256*256 - 1) AS "77777777";
|
||||||
|
77777777
|
||||||
|
----------
|
||||||
|
77777777
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_oct(256::bigint*256::bigint*256::bigint*256::bigint - 1) AS "37777777777";
|
||||||
|
37777777777
|
||||||
|
-------------
|
||||||
|
37777777777
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_hex(-1234) AS "fffffb2e";
|
||||||
|
fffffb2e
|
||||||
|
----------
|
||||||
|
fffffb2e
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select to_hex(-1234::bigint) AS "fffffffffffffb2e";
|
||||||
|
fffffffffffffb2e
|
||||||
|
------------------
|
||||||
|
fffffffffffffb2e
|
||||||
|
(1 row)
|
||||||
|
|
||||||
select to_hex(256*256*256 - 1) AS "ffffff";
|
select to_hex(256*256*256 - 1) AS "ffffff";
|
||||||
ffffff
|
ffffff
|
||||||
--------
|
--------
|
||||||
|
@ -685,10 +685,21 @@ select split_part('joeuser@mydatabase','@',-3) AS "empty string";
|
|||||||
select split_part('@joeuser@mydatabase@','@',-2) AS "mydatabase";
|
select split_part('@joeuser@mydatabase@','@',-2) AS "mydatabase";
|
||||||
|
|
||||||
--
|
--
|
||||||
-- test to_hex
|
-- test to_bin, to_oct, and to_hex
|
||||||
--
|
--
|
||||||
select to_hex(256*256*256 - 1) AS "ffffff";
|
select to_bin(-1234) AS "11111111111111111111101100101110";
|
||||||
|
select to_bin(-1234::bigint);
|
||||||
|
select to_bin(256*256*256 - 1) AS "111111111111111111111111";
|
||||||
|
select to_bin(256::bigint*256::bigint*256::bigint*256::bigint - 1) AS "11111111111111111111111111111111";
|
||||||
|
|
||||||
|
select to_oct(-1234) AS "37777775456";
|
||||||
|
select to_oct(-1234::bigint) AS "1777777777777777775456";
|
||||||
|
select to_oct(256*256*256 - 1) AS "77777777";
|
||||||
|
select to_oct(256::bigint*256::bigint*256::bigint*256::bigint - 1) AS "37777777777";
|
||||||
|
|
||||||
|
select to_hex(-1234) AS "fffffb2e";
|
||||||
|
select to_hex(-1234::bigint) AS "fffffffffffffb2e";
|
||||||
|
select to_hex(256*256*256 - 1) AS "ffffff";
|
||||||
select to_hex(256::bigint*256::bigint*256::bigint*256::bigint - 1) AS "ffffffff";
|
select to_hex(256::bigint*256::bigint*256::bigint*256::bigint - 1) AS "ffffffff";
|
||||||
|
|
||||||
--
|
--
|
||||||
|
Loading…
x
Reference in New Issue
Block a user