2023-08-28 23:46:50 -04:00
|
|
|
#ifdef GEN_INTELLISENSE_DIRECTIVES
|
|
|
|
# pragma once
|
|
|
|
# include "debug.cpp"
|
|
|
|
#endif
|
2023-08-21 20:30:13 -04:00
|
|
|
|
2024-12-13 15:42:13 -05:00
|
|
|
#pragma region String Ops
|
2023-07-25 23:00:57 -04:00
|
|
|
|
|
|
|
internal
|
2024-10-27 18:58:37 -04:00
|
|
|
ssize _scan_zpl_i64( const char* text, s32 base, s64* value )
|
2023-07-24 18:35:16 -04:00
|
|
|
{
|
|
|
|
const char* text_begin = text;
|
|
|
|
s64 result = 0;
|
|
|
|
b32 negative = false;
|
|
|
|
|
|
|
|
if ( *text == '-' )
|
|
|
|
{
|
|
|
|
negative = true;
|
|
|
|
text++;
|
|
|
|
}
|
|
|
|
|
2024-12-12 12:55:15 -05:00
|
|
|
if ( base == 16 && c_str_compare_len( text, "0x", 2 ) == 0 )
|
2023-07-24 18:35:16 -04:00
|
|
|
text += 2;
|
|
|
|
|
|
|
|
for ( ;; )
|
|
|
|
{
|
|
|
|
s64 v;
|
|
|
|
if ( char_is_digit( *text ) )
|
|
|
|
v = *text - '0';
|
|
|
|
else if ( base == 16 && char_is_hex_digit( *text ) )
|
|
|
|
v = hex_digit_to_int( *text );
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
|
|
|
|
result *= base;
|
|
|
|
result += v;
|
|
|
|
text++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( value )
|
|
|
|
{
|
|
|
|
if ( negative )
|
|
|
|
result = -result;
|
|
|
|
*value = result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ( text - text_begin );
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO : Are these good enough for characters?
|
|
|
|
global const char _num_to_char_table[] =
|
|
|
|
"0123456789"
|
|
|
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
|
|
"abcdefghijklmnopqrstuvwxyz"
|
|
|
|
"@$";
|
|
|
|
|
2024-12-12 12:55:15 -05:00
|
|
|
s64 c_str_to_i64( const char* str, char** end_ptr, s32 base )
|
2023-07-24 18:35:16 -04:00
|
|
|
{
|
2024-10-27 18:58:37 -04:00
|
|
|
ssize len;
|
2023-07-24 18:35:16 -04:00
|
|
|
s64 value;
|
|
|
|
|
|
|
|
if ( ! base )
|
|
|
|
{
|
2024-12-12 12:55:15 -05:00
|
|
|
if ( ( c_str_len( str ) > 2 ) && ( c_str_compare_len( str, "0x", 2 ) == 0 ) )
|
2023-07-24 18:35:16 -04:00
|
|
|
base = 16;
|
|
|
|
else
|
|
|
|
base = 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = _scan_zpl_i64( str, base, &value );
|
|
|
|
if ( end_ptr )
|
|
|
|
*end_ptr = ( char* )str + len;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void i64_to_str( s64 value, char* string, s32 base )
|
|
|
|
{
|
|
|
|
char* buf = string;
|
|
|
|
b32 negative = false;
|
|
|
|
u64 v;
|
|
|
|
|
|
|
|
if ( value < 0 )
|
|
|
|
{
|
|
|
|
negative = true;
|
|
|
|
value = -value;
|
|
|
|
}
|
|
|
|
|
2024-10-27 18:58:37 -04:00
|
|
|
v = scast( u64, value);
|
2023-07-24 18:35:16 -04:00
|
|
|
if ( v != 0 )
|
|
|
|
{
|
|
|
|
while ( v > 0 )
|
|
|
|
{
|
|
|
|
*buf++ = _num_to_char_table[ v % base ];
|
|
|
|
v /= base;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*buf++ = '0';
|
|
|
|
}
|
|
|
|
if ( negative )
|
|
|
|
*buf++ = '-';
|
|
|
|
*buf = '\0';
|
2024-12-12 12:55:15 -05:00
|
|
|
c_str_reverse( string );
|
2023-07-24 18:35:16 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void u64_to_str( u64 value, char* string, s32 base )
|
|
|
|
{
|
|
|
|
char* buf = string;
|
|
|
|
|
|
|
|
if ( value )
|
|
|
|
{
|
|
|
|
while ( value > 0 )
|
|
|
|
{
|
|
|
|
*buf++ = _num_to_char_table[ value % base ];
|
|
|
|
value /= base;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*buf++ = '0';
|
|
|
|
}
|
|
|
|
*buf = '\0';
|
|
|
|
|
2024-12-12 12:55:15 -05:00
|
|
|
c_str_reverse( string );
|
2023-07-24 18:35:16 -04:00
|
|
|
}
|
|
|
|
|
2024-12-12 12:55:15 -05:00
|
|
|
f64 c_str_to_f64( const char* str, char** end_ptr )
|
2023-07-24 18:35:16 -04:00
|
|
|
{
|
|
|
|
f64 result, value, sign, scale;
|
|
|
|
s32 frac;
|
|
|
|
|
|
|
|
while ( char_is_space( *str ) )
|
|
|
|
{
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
|
|
|
|
sign = 1.0;
|
|
|
|
if ( *str == '-' )
|
|
|
|
{
|
|
|
|
sign = -1.0;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
else if ( *str == '+' )
|
|
|
|
{
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for ( value = 0.0; char_is_digit( *str ); str++ )
|
|
|
|
{
|
|
|
|
value = value * 10.0 + ( *str - '0' );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( *str == '.' )
|
|
|
|
{
|
|
|
|
f64 pow10 = 10.0;
|
|
|
|
str++;
|
|
|
|
while ( char_is_digit( *str ) )
|
|
|
|
{
|
|
|
|
value += ( *str - '0' ) / pow10;
|
|
|
|
pow10 *= 10.0;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
frac = 0;
|
|
|
|
scale = 1.0;
|
|
|
|
if ( ( *str == 'e' ) || ( *str == 'E' ) )
|
|
|
|
{
|
|
|
|
u32 exp;
|
|
|
|
|
|
|
|
str++;
|
|
|
|
if ( *str == '-' )
|
|
|
|
{
|
|
|
|
frac = 1;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
else if ( *str == '+' )
|
|
|
|
{
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for ( exp = 0; char_is_digit( *str ); str++ )
|
|
|
|
{
|
|
|
|
exp = exp * 10 + ( *str - '0' );
|
|
|
|
}
|
|
|
|
if ( exp > 308 )
|
|
|
|
exp = 308;
|
|
|
|
|
|
|
|
while ( exp >= 50 )
|
|
|
|
{
|
|
|
|
scale *= 1e50;
|
|
|
|
exp -= 50;
|
|
|
|
}
|
|
|
|
while ( exp >= 8 )
|
|
|
|
{
|
|
|
|
scale *= 1e8;
|
|
|
|
exp -= 8;
|
|
|
|
}
|
|
|
|
while ( exp > 0 )
|
|
|
|
{
|
|
|
|
scale *= 10.0;
|
|
|
|
exp -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
result = sign * ( frac ? ( value / scale ) : ( value * scale ) );
|
|
|
|
|
|
|
|
if ( end_ptr )
|
2024-10-27 18:58:37 -04:00
|
|
|
* end_ptr = rcast( char*, ccast(char*, str) );
|
2023-07-24 18:35:16 -04:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2023-07-25 23:00:57 -04:00
|
|
|
|
2024-12-13 15:42:13 -05:00
|
|
|
#pragma endregion String Ops
|