[365] | 1 | // Copyright (C) 2015 ChaosForge Ltd
|
---|
| 2 | // http://chaosforge.org/
|
---|
| 3 | //
|
---|
[395] | 4 | // This file is part of Nova libraries.
|
---|
| 5 | // For conditions of distribution and use, see copying.txt file in root folder.
|
---|
[365] | 6 | //
|
---|
| 7 | // TODO: speedup conversion by doing divisions by 100
|
---|
| 8 |
|
---|
[368] | 9 | #include "nv/stl/string.hh"
|
---|
[365] | 10 |
|
---|
| 11 | #include <cstdio>
|
---|
| 12 | #include <cstdlib>
|
---|
| 13 |
|
---|
| 14 | using namespace nv;
|
---|
| 15 |
|
---|
[402] | 16 | //static const double s_power_10[] = { 1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000 };
|
---|
[365] | 17 |
|
---|
| 18 | static inline void string_reverse( char* begin, char* end )
|
---|
| 19 | {
|
---|
| 20 | char temp;
|
---|
| 21 | while ( end > begin )
|
---|
| 22 | {
|
---|
| 23 | temp = *end;
|
---|
| 24 | *end-- = *begin;
|
---|
| 25 | *begin++ = temp;
|
---|
| 26 | }
|
---|
| 27 | }
|
---|
| 28 |
|
---|
[435] | 29 | nv::size_t nv::sint32_to_buffer( array_ref< char > buffer, sint32 n )
|
---|
[365] | 30 | {
|
---|
[435] | 31 | if ( buffer.size() < 2 ) return 0;
|
---|
| 32 | char* last = buffer.end() - 1;
|
---|
| 33 | char* s = buffer.begin();
|
---|
| 34 | uint32 abs = static_cast<uint32>( n < 0 ? -n : n );
|
---|
[365] | 35 | do
|
---|
| 36 | {
|
---|
[435] | 37 | if ( s == last ) { *buffer.begin() = '\0'; return 0; }
|
---|
[402] | 38 | *s++ = static_cast<char>( '0' + ( abs % 10 ) );
|
---|
[365] | 39 | abs /= 10;
|
---|
| 40 | } while ( abs > 0 );
|
---|
[435] | 41 | if ( n < 0 )
|
---|
| 42 | {
|
---|
| 43 | if ( s == last ) { *buffer.begin() = '\0'; return 0; }
|
---|
| 44 | *s++ = '-';
|
---|
| 45 | }
|
---|
[365] | 46 | *s = '\0';
|
---|
[435] | 47 | string_reverse( buffer.begin(), s - 1 );
|
---|
| 48 | return static_cast<nv::size_t>( s - buffer.begin() );
|
---|
[365] | 49 | }
|
---|
| 50 |
|
---|
[435] | 51 | nv::size_t nv::sint64_to_buffer( array_ref< char > buffer, sint64 n )
|
---|
[365] | 52 | {
|
---|
[435] | 53 | if ( buffer.size() < 2 ) return 0;
|
---|
| 54 | char* last = buffer.end() - 1;
|
---|
| 55 | char* s = buffer.begin();
|
---|
| 56 | uint64 abs = static_cast<uint64>( n < 0 ? -n : n );
|
---|
[365] | 57 | do
|
---|
| 58 | {
|
---|
[435] | 59 | if ( s == last ) { *buffer.begin() = '\0'; return 0; }
|
---|
[402] | 60 | *s++ = static_cast<char>( '0' + ( abs % 10 ) );
|
---|
[365] | 61 | abs /= 10;
|
---|
| 62 | } while ( abs > 0 );
|
---|
[435] | 63 | if ( n < 0 )
|
---|
| 64 | {
|
---|
| 65 | if ( s == last ) { *buffer.begin() = '\0'; return 0; }
|
---|
| 66 | *s++ = '-';
|
---|
| 67 | }
|
---|
[365] | 68 | *s = '\0';
|
---|
[435] | 69 | string_reverse( buffer.begin(), s - 1 );
|
---|
| 70 | return static_cast<nv::size_t>( s - buffer.begin() );
|
---|
[365] | 71 | }
|
---|
| 72 |
|
---|
[435] | 73 | nv::size_t nv::uint32_to_buffer( array_ref< char > buffer, uint32 n )
|
---|
[365] | 74 | {
|
---|
[435] | 75 | if ( buffer.size() < 2 ) return 0;
|
---|
| 76 | char* last = buffer.end() - 1;
|
---|
| 77 | char* s = buffer.begin();
|
---|
[365] | 78 | do
|
---|
| 79 | {
|
---|
[435] | 80 | if ( s == last ) { *buffer.begin() = '\0'; return 0; }
|
---|
[402] | 81 | *s++ = static_cast<char>( '0' + ( n % 10 ) );
|
---|
[365] | 82 | n /= 10;
|
---|
| 83 | } while ( n > 0 );
|
---|
| 84 | *s = '\0';
|
---|
[435] | 85 | string_reverse( buffer.begin(), s - 1 );
|
---|
| 86 | return static_cast<nv::size_t>( s - buffer.begin() );
|
---|
[365] | 87 | }
|
---|
| 88 |
|
---|
[435] | 89 | nv::size_t nv::uint64_to_buffer( array_ref< char > buffer, uint64 n )
|
---|
[365] | 90 | {
|
---|
[435] | 91 | if ( buffer.size() < 2 ) return 0;
|
---|
| 92 | char* last = buffer.end() - 1;
|
---|
| 93 | char* s = buffer.begin();
|
---|
[365] | 94 | do
|
---|
| 95 | {
|
---|
[435] | 96 | if ( s == last ) { *buffer.begin() = '\0'; return 0; }
|
---|
[402] | 97 | *s++ = static_cast<char>( '0' + ( n % 10 ) );
|
---|
[365] | 98 | n /= 10;
|
---|
| 99 | } while ( n > 0 );
|
---|
| 100 | *s = '\0';
|
---|
[435] | 101 | string_reverse( buffer.begin(), s - 1 );
|
---|
| 102 | return static_cast<nv::size_t>( s - buffer.begin() );
|
---|
[365] | 103 | }
|
---|
| 104 |
|
---|
[435] | 105 | nv::size_t nv::f32_to_buffer( array_ref< char > buffer, f32 n )
|
---|
[365] | 106 | {
|
---|
| 107 | #if NV_COMPILER == NV_MSVC
|
---|
[435] | 108 | int result = sprintf_s( buffer.data(), buffer.size(), "%.*g", 6, n );
|
---|
[365] | 109 | #else
|
---|
[435] | 110 | int result = snprintf( buffer.data(), buffer.size(), "%.*g", 6, n );
|
---|
[365] | 111 | #endif
|
---|
[402] | 112 | return static_cast<nv::size_t>( result > 0 ? result : 0 );
|
---|
[365] | 113 | }
|
---|
| 114 |
|
---|
[435] | 115 | nv::size_t nv::f64_to_buffer( array_ref< char > buffer, f64 n )
|
---|
[365] | 116 | {
|
---|
| 117 | #if NV_COMPILER == NV_MSVC
|
---|
[435] | 118 | int result = sprintf_s( buffer.data(), buffer.size(), "%.*g", 6, n );
|
---|
[365] | 119 | #else
|
---|
[435] | 120 | int result = snprintf( buffer.data(), buffer.size(), "%.*g", 6, n );
|
---|
[365] | 121 | #endif
|
---|
[402] | 122 | return static_cast<nv::size_t>( result > 0 ? result : 0 );
|
---|
[365] | 123 | }
|
---|
| 124 |
|
---|
[402] | 125 | sint32 nv::buffer_to_sint32( const char* str, char** end )
|
---|
[365] | 126 | {
|
---|
| 127 | const char* s = str;
|
---|
| 128 | while ( *s == ' ' ) ++s;
|
---|
| 129 | sint32 result = 0;
|
---|
| 130 | bool positive = true;
|
---|
| 131 | switch ( *s )
|
---|
| 132 | {
|
---|
| 133 | case '-': ++s; positive = false; break;
|
---|
| 134 | case '+': ++s; break;
|
---|
| 135 | default: break;
|
---|
| 136 | }
|
---|
| 137 | while ( *s >= '0' && *s <= '9' )
|
---|
| 138 | {
|
---|
| 139 | result = ( result * 10 ) + ( *s - '0' );
|
---|
| 140 | ++s;
|
---|
| 141 | }
|
---|
[402] | 142 | if ( end != nullptr ) *end = const_cast<char*>( s );
|
---|
[365] | 143 | return positive ? result : -result;
|
---|
| 144 | }
|
---|
| 145 |
|
---|
[402] | 146 | sint64 nv::buffer_to_sint64( const char* s, char** end )
|
---|
[365] | 147 | {
|
---|
| 148 | while ( *s == ' ' ) ++s;
|
---|
| 149 | sint64 result = 0;
|
---|
| 150 | bool positive = true;
|
---|
| 151 | switch ( *s )
|
---|
| 152 | {
|
---|
| 153 | case '-': ++s; positive = false; break;
|
---|
| 154 | case '+': ++s; break;
|
---|
| 155 | default: break;
|
---|
| 156 | }
|
---|
| 157 | while ( *s >= '0' && *s <= '9' )
|
---|
| 158 | {
|
---|
| 159 | result = ( result * 10 ) + ( *s - '0' );
|
---|
| 160 | ++s;
|
---|
| 161 | }
|
---|
[402] | 162 | if ( end != nullptr ) *end = const_cast<char*>( s );
|
---|
[365] | 163 | return positive ? result : -result;
|
---|
| 164 | }
|
---|
| 165 |
|
---|
[402] | 166 | uint32 nv::buffer_to_uint32( const char* s, char** end )
|
---|
[365] | 167 | {
|
---|
| 168 | while ( *s == ' ' ) ++s;
|
---|
| 169 | uint32 result = 0;
|
---|
| 170 | while ( *s >= '0' && *s <= '9' )
|
---|
| 171 | {
|
---|
[402] | 172 | result = ( result * 10 ) + static_cast<uint32>( *s - '0' );
|
---|
[365] | 173 | ++s;
|
---|
| 174 | }
|
---|
[402] | 175 | if ( end != nullptr ) *end = const_cast<char*>( s );
|
---|
[365] | 176 | return result;
|
---|
| 177 | }
|
---|
| 178 |
|
---|
[402] | 179 | uint64 nv::buffer_to_uint64( const char* s, char** end )
|
---|
[365] | 180 | {
|
---|
| 181 | while ( *s == ' ' ) ++s;
|
---|
| 182 | uint64 result = 0;
|
---|
| 183 | while ( *s >= '0' && *s <= '9' )
|
---|
| 184 | {
|
---|
[402] | 185 | result = ( result * 10 ) + static_cast<uint32>( *s - '0' );
|
---|
[365] | 186 | ++s;
|
---|
| 187 | }
|
---|
[402] | 188 | if ( end != nullptr ) *end = const_cast<char*>( s );
|
---|
[365] | 189 | return result;
|
---|
| 190 | }
|
---|
| 191 |
|
---|
[402] | 192 | float nv::buffer_to_f32( const char* s, char** end )
|
---|
[365] | 193 | {
|
---|
| 194 | return strtof( s, end );
|
---|
| 195 | }
|
---|
| 196 |
|
---|
[402] | 197 | double nv::buffer_to_f64( const char* s, char** end )
|
---|
[365] | 198 | {
|
---|
| 199 | return strtod( s, end );
|
---|
| 200 | }
|
---|