306 lines
9.6 KiB
C
306 lines
9.6 KiB
C
|
/****************************************************************
|
||
|
* *
|
||
|
* Copyright 2001, 2011 Fidelity Information Services, Inc *
|
||
|
* *
|
||
|
* This source code contains the intellectual property *
|
||
|
* of its copyright holder(s), and is made available *
|
||
|
* under a license. If you do not know the terms of *
|
||
|
* the license, please stop and do not read further. *
|
||
|
* *
|
||
|
****************************************************************/
|
||
|
|
||
|
#include "mdef.h"
|
||
|
|
||
|
#include "arit.h"
|
||
|
#include "gdsroot.h"
|
||
|
#include "gtm_facility.h"
|
||
|
#include "fileinfo.h"
|
||
|
#include "gdsbt.h"
|
||
|
#include "gdsfhead.h"
|
||
|
#include "collseq.h"
|
||
|
#include "copy.h"
|
||
|
#include "stringpool.h"
|
||
|
#include "do_xform.h"
|
||
|
#include "format_targ_key.h"
|
||
|
|
||
|
GBLREF gv_namehead *gv_target;
|
||
|
GBLREF gd_region *gv_cur_region;
|
||
|
|
||
|
static readonly unsigned char pos_code[100] =
|
||
|
{
|
||
|
0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a,
|
||
|
0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a,
|
||
|
0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a,
|
||
|
0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a,
|
||
|
0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4a,
|
||
|
0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5a,
|
||
|
0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6a,
|
||
|
0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7a,
|
||
|
0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8a,
|
||
|
0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9a
|
||
|
};
|
||
|
|
||
|
static readonly unsigned char neg_code[100] =
|
||
|
{
|
||
|
0xfe, 0xfd, 0xfc, 0xfb, 0xfa, 0xf9, 0xf8, 0xf7, 0xf6, 0xf5,
|
||
|
0xee, 0xed, 0xec, 0xeb, 0xea, 0xe9, 0xe8, 0xe7, 0xe6, 0xe5,
|
||
|
0xde, 0xdd, 0xdc, 0xdb, 0xda, 0xd9, 0xd8, 0xd7, 0xd6, 0xd5,
|
||
|
0xce, 0xcd, 0xcc, 0xcb, 0xca, 0xc9, 0xc8, 0xc7, 0xc6, 0xc5,
|
||
|
0xbe, 0xbd, 0xbc, 0xbb, 0xba, 0xb9, 0xb8, 0xb7, 0xb6, 0xb5,
|
||
|
0xae, 0xad, 0xac, 0xab, 0xaa, 0xa9, 0xa8, 0xa7, 0xa6, 0xa5,
|
||
|
0x9e, 0x9d, 0x9c, 0x9b, 0x9a, 0x99, 0x98, 0x97, 0x96, 0x95,
|
||
|
0x8e, 0x8d, 0x8c, 0x8b, 0x8a, 0x89, 0x88, 0x87, 0x86, 0x85,
|
||
|
0x7e, 0x7d, 0x7c, 0x7b, 0x7a, 0x79, 0x78, 0x77, 0x76, 0x75,
|
||
|
0x6e, 0x6d, 0x6c, 0x6b, 0x6a, 0x69, 0x68, 0x67, 0x66, 0x65
|
||
|
};
|
||
|
|
||
|
unsigned char *mval2subsc(mval *in_val, gv_key *out_key)
|
||
|
{
|
||
|
boolean_t is_negative;
|
||
|
unsigned char buf1[MAX_KEY_SZ + 1], ch, *cvt_table, *in_ptr, *out_ptr;
|
||
|
unsigned char *tm, temp_mantissa[NUM_DEC_DG_2L / 2 + 3]; /* Need 1 byte for each two digits. Add 3 bytes slop */
|
||
|
mstr mstr_ch, mstr_buf1;
|
||
|
int4 mt, mw, mx;
|
||
|
uint4 mvt; /* Local copy of mvtype, bit ands use a int4, so do conversion once */
|
||
|
unsigned int digs, exp_val;
|
||
|
int tmp_len, avail_bytes;
|
||
|
DCL_THREADGBL_ACCESS;
|
||
|
|
||
|
SETUP_THREADGBL_ACCESS;
|
||
|
/* The below assert is an attempt to catch between some and many of the cases where the mvtype is
|
||
|
* not an accurate representation of the content. This can happen (and has) when there is a bug in
|
||
|
* op_svput() where only the string portion of an mval is set without (re)setting the type leading
|
||
|
* to numeric or num-approx values that do not represent reality and causing trouble especially with
|
||
|
* subscripts. In that example, "-1" was left as a string with MV_NUM_APPROX on which caused a lot
|
||
|
* of trouble with $ORDER in a database when -1 was treated as a string. This assert is not a 100%
|
||
|
* catchall of invalid settings but it provides at least some barrier. A full barrier would require
|
||
|
* complete conversion which is a bit expensive to always re-do at this point - even in a dbg version.
|
||
|
*/
|
||
|
assert(!(MV_NUM_APPROX & in_val->mvtype) || (NUM_DEC_DG_2L < in_val->str.len) || !val_iscan(in_val));
|
||
|
out_ptr = out_key->base + out_key->end;
|
||
|
if (TREF(transform) && gv_target->nct)
|
||
|
{
|
||
|
MV_FORCE_STR(in_val);
|
||
|
mvt = in_val->mvtype | MV_NUM_APPROX;
|
||
|
} else
|
||
|
mvt = (uint4)in_val->mvtype;
|
||
|
if (!(mvt & (MV_NM | MV_NUM_APPROX)))
|
||
|
{ /* Not currently in numeric form. Is it cannonical? */
|
||
|
if (val_iscan(in_val))
|
||
|
{ /* Yes, convert it to numeric */
|
||
|
(void)s2n(in_val);
|
||
|
mvt = in_val->mvtype;
|
||
|
assert(mvt & MV_NM);
|
||
|
} else
|
||
|
{ /* No, not numeric. Note the fact for future reference */
|
||
|
mvt = in_val->mvtype |= MV_NUM_APPROX;
|
||
|
}
|
||
|
}
|
||
|
if (mvt & MV_NUM_APPROX)
|
||
|
{ /* It's a string */
|
||
|
in_ptr = (unsigned char *)in_val->str.addr;
|
||
|
tmp_len = in_val->str.len;
|
||
|
if (TREF(transform) && gv_target->collseq)
|
||
|
{
|
||
|
mstr_ch.len = tmp_len;
|
||
|
mstr_ch.addr = (char *)in_ptr;
|
||
|
mstr_buf1.len = SIZEOF(buf1);
|
||
|
mstr_buf1.addr = (char *)buf1;
|
||
|
do_xform(gv_target->collseq, XFORM, &mstr_ch, &mstr_buf1, &tmp_len);
|
||
|
in_ptr = (unsigned char*)mstr_buf1.addr; /* mstr_buf1.addr is used just in case it is
|
||
|
reallocated by the XFORM routine */
|
||
|
}
|
||
|
/* Find out how much space is needed at a minimum to store the subscript representation of this string.
|
||
|
* That would be STR_SUB_PREFIX + input string + at most TWO KEY_DELIMITER.
|
||
|
* Assuming this, compute how much space would still be available in the out_key before reaching the top.
|
||
|
* If this is negative, we have to signal a GVSUBOFLOW error.
|
||
|
* If this is positive and the input string contains 0x00 or 0x01, we would need additional bytes to
|
||
|
* store the STR_SUB_ESCAPE byte. Decrement the available space until it becomes zero
|
||
|
* at which point issue a GVSUBOFLOW error as well.
|
||
|
*/
|
||
|
avail_bytes = (out_key->top - MAX_NUM_SUBSC_LEN) - (out_key->end + tmp_len + 3);
|
||
|
if (0 > avail_bytes)
|
||
|
ISSUE_GVSUBOFLOW_ERROR(out_key);
|
||
|
if (0 < tmp_len)
|
||
|
{
|
||
|
*out_ptr++ = STR_SUB_PREFIX;
|
||
|
do
|
||
|
{
|
||
|
ch = *in_ptr++;
|
||
|
if (ch <= 1)
|
||
|
{
|
||
|
*out_ptr++ = STR_SUB_ESCAPE;
|
||
|
if (0 > --avail_bytes)
|
||
|
{
|
||
|
/* Ensure input key to format_targ_key is double null terminated */
|
||
|
assert(STR_SUB_PREFIX == out_key->base[out_key->end]);
|
||
|
out_key->base[out_key->end] = KEY_DELIMITER;
|
||
|
ISSUE_GVSUBOFLOW_ERROR(out_key);
|
||
|
}
|
||
|
ch++; /* promote character */
|
||
|
}
|
||
|
*out_ptr++ = ch;
|
||
|
} while (--tmp_len > 0);
|
||
|
} else
|
||
|
{
|
||
|
*out_ptr++ = (!TREF(transform) || (0 == gv_cur_region->std_null_coll))
|
||
|
? STR_SUB_PREFIX : SUBSCRIPT_STDCOL_NULL;
|
||
|
}
|
||
|
goto FINI;
|
||
|
}
|
||
|
/* Its a number, is it an integer? But before this assert that we have enough allocated space in the key
|
||
|
* to store the maximum possible numeric subscript and two terminating 0s at the end of the key */
|
||
|
assert((MAX_NUM_SUBSC_LEN + 2) <= (int)(out_key->top - out_key->end));
|
||
|
if (mvt & MV_INT)
|
||
|
{ /* Yes, its an integer, convert it */
|
||
|
is_negative = FALSE;
|
||
|
cvt_table = pos_code;
|
||
|
if (0 > (mt = in_val->m[1]))
|
||
|
{
|
||
|
is_negative = TRUE;
|
||
|
cvt_table = neg_code;
|
||
|
mt = -mt;
|
||
|
} else if (0 == mt)
|
||
|
{
|
||
|
*out_ptr++ = SUBSCRIPT_ZERO;
|
||
|
goto FINI;
|
||
|
}
|
||
|
if (10 > mt)
|
||
|
{
|
||
|
*out_ptr++ = is_negative ? ~(SUBSCRIPT_BIAS - 2) : (SUBSCRIPT_BIAS - 2);
|
||
|
*out_ptr++ = cvt_table[mt * 10];
|
||
|
goto FINISH_NUMBER;
|
||
|
}
|
||
|
if (100 > mt)
|
||
|
{
|
||
|
*out_ptr++ = is_negative ? ~(SUBSCRIPT_BIAS - 1) : (SUBSCRIPT_BIAS - 1);
|
||
|
*out_ptr++ = cvt_table[mt];
|
||
|
goto FINISH_NUMBER;
|
||
|
}
|
||
|
tm = temp_mantissa;
|
||
|
if (1000 > mt)
|
||
|
{
|
||
|
exp_val = SUBSCRIPT_BIAS;
|
||
|
goto ODD_INTEGER;
|
||
|
}
|
||
|
if (10000 > mt)
|
||
|
{
|
||
|
exp_val = SUBSCRIPT_BIAS + 1;
|
||
|
goto EVEN_INTEGER;
|
||
|
}
|
||
|
if (100000 > mt)
|
||
|
{
|
||
|
exp_val = SUBSCRIPT_BIAS + 2;
|
||
|
goto ODD_INTEGER;
|
||
|
}
|
||
|
if (1000000 > mt)
|
||
|
{
|
||
|
exp_val = SUBSCRIPT_BIAS + 3;
|
||
|
goto EVEN_INTEGER;
|
||
|
}
|
||
|
if (10000000 > mt)
|
||
|
{
|
||
|
exp_val = SUBSCRIPT_BIAS + 4;
|
||
|
goto ODD_INTEGER;
|
||
|
}
|
||
|
if (100000000 > mt)
|
||
|
{
|
||
|
exp_val = SUBSCRIPT_BIAS + 5;
|
||
|
goto EVEN_INTEGER;
|
||
|
}
|
||
|
exp_val = SUBSCRIPT_BIAS + 6;
|
||
|
ODD_INTEGER:
|
||
|
*out_ptr++ = is_negative ? ~(exp_val) : (exp_val);
|
||
|
mw = mx = mt / 10;
|
||
|
mw *= 10;
|
||
|
mw = mt - mw;
|
||
|
mt = mx;
|
||
|
if (mw)
|
||
|
{
|
||
|
*tm++ = cvt_table[mw * 10];
|
||
|
goto FINISH_INTEGERS;
|
||
|
}
|
||
|
goto KEEP_STRIPING;
|
||
|
EVEN_INTEGER:
|
||
|
*out_ptr++ = is_negative ? ~(exp_val) : (exp_val);
|
||
|
KEEP_STRIPING:
|
||
|
while (mt)
|
||
|
{
|
||
|
mw = mx = mt / 100;
|
||
|
mw *= 100;
|
||
|
mw = mt - mw;
|
||
|
mt = mx;
|
||
|
if (mw)
|
||
|
{
|
||
|
*tm++ = cvt_table[mw];
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
FINISH_INTEGERS:
|
||
|
while (mt)
|
||
|
{
|
||
|
mw = mx = mt / 100;
|
||
|
mw *= 100;
|
||
|
mw = mt - mw;
|
||
|
*tm++ = cvt_table[mw];
|
||
|
mt = mx;
|
||
|
}
|
||
|
while (tm > temp_mantissa)
|
||
|
*out_ptr++ = *--tm;
|
||
|
goto FINISH_NUMBER;
|
||
|
}
|
||
|
/* Convert 18 digit number */
|
||
|
cvt_table = pos_code;
|
||
|
if (0 != (is_negative = in_val->sgn))
|
||
|
cvt_table = neg_code;
|
||
|
*out_ptr++ = is_negative ? ~(in_val->e - MV_XBIAS + SUBSCRIPT_BIAS) : (in_val->e - MV_XBIAS + SUBSCRIPT_BIAS);
|
||
|
mt = in_val->m[1];
|
||
|
mw = in_val->m[0];
|
||
|
/* Strip top two digits */
|
||
|
mx = mt / (MANT_HI / 100);
|
||
|
*out_ptr++ = cvt_table[mx];
|
||
|
mt = (mt - (mx * (MANT_HI / 100))) * 100;
|
||
|
/*
|
||
|
* The two msd's have now been converted. The maximum number of
|
||
|
* data remaining is 7 digits in "mt" and 9 digits in "mw".
|
||
|
* If mw is zero, then we should just grind out mt till we are done
|
||
|
*/
|
||
|
if (0 == mw)
|
||
|
goto LAST_LONGWORD;
|
||
|
/* there are more than 7 digits left. First, we will put 8 digits in mt, (leaving 8 digits in mw) */
|
||
|
mx = mw / (MANT_HI / 10);
|
||
|
mt += mx * 10;
|
||
|
mw = (mw - (mx * (MANT_HI / 10))) * 10;
|
||
|
if (0 == mw)
|
||
|
goto LAST_LONGWORD;
|
||
|
for (digs = 0; digs < 4; digs++)
|
||
|
{
|
||
|
mx = mt / (MANT_HI / 100);
|
||
|
*out_ptr++ = cvt_table[mx];
|
||
|
mt = (mt - (mx * (MANT_HI / 100))) * 100;
|
||
|
}
|
||
|
mt = mw;
|
||
|
LAST_LONGWORD:
|
||
|
while (mt)
|
||
|
{
|
||
|
mx = mt / (MANT_HI / 100);
|
||
|
*out_ptr++ = cvt_table[mx];
|
||
|
mt = (mt - (mx * (MANT_HI / 100))) * 100;
|
||
|
}
|
||
|
FINISH_NUMBER:
|
||
|
if (is_negative)
|
||
|
*out_ptr++ = NEG_MNTSSA_END;
|
||
|
FINI:
|
||
|
*out_ptr++ = KEY_DELIMITER;
|
||
|
*out_ptr = KEY_DELIMITER;
|
||
|
out_key->prev = out_key->end;
|
||
|
out_key->end = out_ptr - out_key->base;
|
||
|
/* Check if after adding the current subscript and the second terminating NULL byte, there is still
|
||
|
* MAX_NUM_SUBSC_LEN bytes (allocated additionally as part of the DBKEYSIZE macro) left at the end.
|
||
|
* If not, we have overflown the original max-key-size length. Issue error.
|
||
|
*/
|
||
|
if ((MAX_NUM_SUBSC_LEN + 1) >= (int)(out_key->top - out_key->end))
|
||
|
ISSUE_GVSUBOFLOW_ERROR(out_key);
|
||
|
return out_ptr;
|
||
|
}
|