VirtualBox

source: vbox/trunk/src/VBox/Runtime/testcase/tstStrToNum.cpp@ 79798

Last change on this file since 79798 was 79562, checked in by vboxsync, 6 years ago

IPRT: Added RTSTRCONVERTHEXBYTES_F_SEP_COLON as well as a RTStrConvertHexBytesEx variant. bugref:9288

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 13.5 KB
Line 
1/* $Id: tstStrToNum.cpp 79562 2019-07-05 20:37:19Z vboxsync $ */
2/** @file
3 * IPRT Testcase - String To Number Conversion.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/test.h>
32#include <iprt/string.h>
33#include <iprt/stream.h>
34#include <iprt/err.h>
35
36struct TstI64
37{
38 const char *psz;
39 unsigned uBase;
40 int rc;
41 int64_t Result;
42};
43
44struct TstU64
45{
46 const char *psz;
47 unsigned uBase;
48 int rc;
49 uint64_t Result;
50};
51
52struct TstI32
53{
54 const char *psz;
55 unsigned uBase;
56 int rc;
57 int32_t Result;
58};
59
60struct TstU32
61{
62 const char *psz;
63 unsigned uBase;
64 int rc;
65 uint32_t Result;
66};
67
68
69#define TEST(Test, Type, Fmt, Fun, iTest) \
70 do \
71 { \
72 Type Result; \
73 int rc = Fun(Test.psz, NULL, Test.uBase, &Result); \
74 if (Result != Test.Result) \
75 RTTestIFailed("'%s' -> " Fmt " expected " Fmt ". (%s/%u)\n", Test.psz, Result, Test.Result, #Fun, iTest); \
76 else if (rc != Test.rc) \
77 RTTestIFailed("'%s' -> rc=%Rrc expected %Rrc. (%s/%u)\n", Test.psz, rc, Test.rc, #Fun, iTest); \
78 } while (0)
79
80
81#define RUN_TESTS(aTests, Type, Fmt, Fun) \
82 do \
83 { \
84 for (unsigned iTest = 0; iTest < RT_ELEMENTS(aTests); iTest++) \
85 { \
86 TEST(aTests[iTest], Type, Fmt, Fun, iTest); \
87 } \
88 } while (0)
89
90int main()
91{
92 RTTEST hTest;
93 RTEXITCODE rcExit = RTTestInitAndCreate("tstRTStrToNum", &hTest);
94 if (rcExit != RTEXITCODE_SUCCESS)
95 return rcExit;
96
97 static const struct TstU64 aTstU64[] =
98 {
99 { "0", 0, VINF_SUCCESS, 0 },
100 { "1", 0, VINF_SUCCESS, 1 },
101 { "-1", 0, VWRN_NEGATIVE_UNSIGNED, ~0ULL },
102 { "0x", 0, VWRN_TRAILING_CHARS, 0 },
103 { "0x1", 0, VINF_SUCCESS, 1 },
104 { "0x0fffffffffffffff", 0, VINF_SUCCESS, 0x0fffffffffffffffULL },
105 { "0x0ffffffffffffffffffffff",0, VWRN_NUMBER_TOO_BIG, 0xffffffffffffffffULL },
106 { "asdfasdfasdf", 0, VERR_NO_DIGITS, 0 },
107 { "0x111111111", 0, VINF_SUCCESS, 0x111111111ULL },
108 { "4D9702C5CBD9B778", 16, VINF_SUCCESS, UINT64_C(0x4D9702C5CBD9B778) },
109 };
110 RUN_TESTS(aTstU64, uint64_t, "%#llx", RTStrToUInt64Ex);
111
112 static const struct TstI64 aTstI64[] =
113 {
114 { "0", 0, VINF_SUCCESS, 0 },
115 { "1", 0, VINF_SUCCESS, 1 },
116 { "-1", 0, VINF_SUCCESS, -1 },
117 { "-1", 10, VINF_SUCCESS, -1 },
118 { "-31", 0, VINF_SUCCESS, -31 },
119 { "-31", 10, VINF_SUCCESS, -31 },
120 { "-32", 0, VINF_SUCCESS, -32 },
121 { "-33", 0, VINF_SUCCESS, -33 },
122 { "-64", 0, VINF_SUCCESS, -64 },
123 { "-127", 0, VINF_SUCCESS, -127 },
124 { "-128", 0, VINF_SUCCESS, -128 },
125 { "-129", 0, VINF_SUCCESS, -129 },
126 { "-254", 0, VINF_SUCCESS, -254 },
127 { "-255", 0, VINF_SUCCESS, -255 },
128 { "-256", 0, VINF_SUCCESS, -256 },
129 { "-257", 0, VINF_SUCCESS, -257 },
130 { "-511", 0, VINF_SUCCESS, -511 },
131 { "-512", 0, VINF_SUCCESS, -512 },
132 { "-513", 0, VINF_SUCCESS, -513 },
133 { "-1023", 0, VINF_SUCCESS, -1023 },
134 { "-1023", 0, VINF_SUCCESS, -1023 },
135 { "-1023", 0, VINF_SUCCESS, -1023},
136 { "-1023", 10, VINF_SUCCESS, -1023 },
137 { "-4564678", 0, VINF_SUCCESS, -4564678 },
138 { "-4564678", 10, VINF_SUCCESS, -4564678 },
139 { "-1234567890123456789", 0, VINF_SUCCESS, -1234567890123456789LL },
140 { "-1234567890123456789", 10, VINF_SUCCESS, -1234567890123456789LL },
141 { "0x", 0, VWRN_TRAILING_CHARS, 0 },
142 { "0x1", 0, VINF_SUCCESS, 1 },
143 { "0x1", 10, VWRN_TRAILING_CHARS, 0 },
144 { "0x1", 16, VINF_SUCCESS, 1 },
145 { "0x0fffffffffffffff", 0, VINF_SUCCESS, 0x0fffffffffffffffULL },
146 { "0x7fffffffffffffff", 0, VINF_SUCCESS, 0x7fffffffffffffffULL },
147 { "0xffffffffffffffff", 0, VWRN_NUMBER_TOO_BIG, -1 },
148 { "0x01111111111111111111111",0, VWRN_NUMBER_TOO_BIG, 0x1111111111111111ULL },
149 { "0x02222222222222222222222",0, VWRN_NUMBER_TOO_BIG, 0x2222222222222222ULL },
150 { "0x03333333333333333333333",0, VWRN_NUMBER_TOO_BIG, 0x3333333333333333ULL },
151 { "0x04444444444444444444444",0, VWRN_NUMBER_TOO_BIG, 0x4444444444444444ULL },
152 { "0x07777777777777777777777",0, VWRN_NUMBER_TOO_BIG, 0x7777777777777777ULL },
153 { "0x07f7f7f7f7f7f7f7f7f7f7f",0, VWRN_NUMBER_TOO_BIG, 0x7f7f7f7f7f7f7f7fULL },
154 { "0x0ffffffffffffffffffffff",0, VWRN_NUMBER_TOO_BIG, (int64_t)0xffffffffffffffffULL },
155 { "asdfasdfasdf", 0, VERR_NO_DIGITS, 0 },
156 { "0x111111111", 0, VINF_SUCCESS, 0x111111111ULL },
157 };
158 RUN_TESTS(aTstI64, int64_t, "%#lld", RTStrToInt64Ex);
159
160
161 static const struct TstI32 aTstI32[] =
162 {
163 { "0", 0, VINF_SUCCESS, 0 },
164 { "1", 0, VINF_SUCCESS, 1 },
165 { "-1", 0, VINF_SUCCESS, -1 },
166 { "-1", 10, VINF_SUCCESS, -1 },
167 { "-31", 0, VINF_SUCCESS, -31 },
168 { "-31", 10, VINF_SUCCESS, -31 },
169 { "-32", 0, VINF_SUCCESS, -32 },
170 { "-33", 0, VINF_SUCCESS, -33 },
171 { "-64", 0, VINF_SUCCESS, -64 },
172 { "-127", 0, VINF_SUCCESS, -127 },
173 { "-128", 0, VINF_SUCCESS, -128 },
174 { "-129", 0, VINF_SUCCESS, -129 },
175 { "-254", 0, VINF_SUCCESS, -254 },
176 { "-255", 0, VINF_SUCCESS, -255 },
177 { "-256", 0, VINF_SUCCESS, -256 },
178 { "-257", 0, VINF_SUCCESS, -257 },
179 { "-511", 0, VINF_SUCCESS, -511 },
180 { "-512", 0, VINF_SUCCESS, -512 },
181 { "-513", 0, VINF_SUCCESS, -513 },
182 { "-1023", 0, VINF_SUCCESS, -1023 },
183 { "-1023", 0, VINF_SUCCESS, -1023 },
184 { "-1023", 0, VINF_SUCCESS, -1023},
185 { "-1023", 10, VINF_SUCCESS, -1023 },
186 { "-4564678", 0, VINF_SUCCESS, -4564678 },
187 { "-4564678", 10, VINF_SUCCESS, -4564678 },
188 { "4564678", 0, VINF_SUCCESS, 4564678 },
189 { "4564678", 10, VINF_SUCCESS, 4564678 },
190 { "-1234567890123456789", 0, VWRN_NUMBER_TOO_BIG, (int32_t)((uint64_t)INT64_C(-1234567890123456789) & UINT32_MAX) },
191 { "-1234567890123456789", 10, VWRN_NUMBER_TOO_BIG, (int32_t)((uint64_t)INT64_C(-1234567890123456789) & UINT32_MAX) },
192 { "1234567890123456789", 0, VWRN_NUMBER_TOO_BIG, (int32_t)(INT64_C(1234567890123456789) & UINT32_MAX) },
193 { "1234567890123456789", 10, VWRN_NUMBER_TOO_BIG, (int32_t)(INT64_C(1234567890123456789) & UINT32_MAX) },
194 { "0x", 0, VWRN_TRAILING_CHARS, 0 },
195 { "0x1", 0, VINF_SUCCESS, 1 },
196 { "0x1", 10, VWRN_TRAILING_CHARS, 0 },
197 { "0x1", 16, VINF_SUCCESS, 1 },
198 { "0x7fffffff", 0, VINF_SUCCESS, 0x7fffffff },
199 { "0x80000000", 0, VWRN_NUMBER_TOO_BIG, INT32_MIN },
200 { "0xffffffff", 0, VWRN_NUMBER_TOO_BIG, -1 },
201 { "0x0fffffffffffffff", 0, VWRN_NUMBER_TOO_BIG, (int32_t)0xffffffff },
202 { "0x01111111111111111111111",0, VWRN_NUMBER_TOO_BIG, 0x11111111 },
203 { "0x0ffffffffffffffffffffff",0, VWRN_NUMBER_TOO_BIG, (int32_t)0xffffffff },
204 { "asdfasdfasdf", 0, VERR_NO_DIGITS, 0 },
205 { "0x1111111", 0, VINF_SUCCESS, 0x01111111 },
206 };
207 RUN_TESTS(aTstI32, int32_t, "%#d", RTStrToInt32Ex);
208
209 static const struct TstU32 aTstU32[] =
210 {
211 { "0", 0, VINF_SUCCESS, 0 },
212 { "1", 0, VINF_SUCCESS, 1 },
213 /// @todo { "-1", 0, VWRN_NEGATIVE_UNSIGNED, ~0 }, - no longer true. bad idea?
214 { "-1", 0, VWRN_NUMBER_TOO_BIG, ~0U },
215 { "0x", 0, VWRN_TRAILING_CHARS, 0 },
216 { "0x1", 0, VINF_SUCCESS, 1 },
217 { "0x0fffffffffffffff", 0, VWRN_NUMBER_TOO_BIG, 0xffffffffU },
218 { "0x0ffffffffffffffffffffff",0, VWRN_NUMBER_TOO_BIG, 0xffffffffU },
219 { "asdfasdfasdf", 0, VERR_NO_DIGITS, 0 },
220 { "0x1111111", 0, VINF_SUCCESS, 0x1111111 },
221 };
222 RUN_TESTS(aTstU32, uint32_t, "%#x", RTStrToUInt32Ex);
223
224
225 /*
226 * Test the some hex stuff too.
227 */
228 static const struct
229 {
230 const char *pszHex;
231 size_t cbOut;
232 size_t offNext;
233 uint8_t bLast;
234 bool fColon;
235 int rc;
236 } s_aConvertHexTests[] =
237 {
238 { "00", 1, 2, 0x00, true, VINF_SUCCESS },
239 { "00", 1, 2, 0x00, false, VINF_SUCCESS },
240 { "000102", 3, 6, 0x02, true, VINF_SUCCESS },
241 { "00019", 2, 4, 0x01, false, VERR_UNEVEN_INPUT },
242 { "00019", 2, 4, 0x01, true, VERR_UNEVEN_INPUT },
243 { "0001:9", 3, 6, 0x09, true, VINF_SUCCESS},
244 { "000102", 3, 6, 0x02, false, VINF_SUCCESS },
245 { "0:1", 2, 3, 0x01, true, VINF_SUCCESS },
246 { ":", 2, 1, 0x00, true, VINF_SUCCESS },
247 { "0:01", 2, 4, 0x01, true, VINF_SUCCESS },
248 { "00:01", 2, 5, 0x01, true, VINF_SUCCESS },
249 { ":1:2:3:4:5", 6, 10, 0x05, true, VINF_SUCCESS },
250 { ":1:2:3::5", 6, 9, 0x05, true, VINF_SUCCESS },
251 { ":1:2:3:4:", 6, 9, 0x00, true, VINF_SUCCESS },
252 };
253 for (unsigned i = 0; i < RT_ELEMENTS(s_aConvertHexTests); i++)
254 {
255 uint8_t abBuf[1024];
256 memset(abBuf, 0xf6, sizeof(abBuf));
257 const char *pszExpectNext = &s_aConvertHexTests[i].pszHex[s_aConvertHexTests[i].offNext];
258 const char *pszNext = "";
259 size_t cbReturned = 77777;
260 int rc = RTStrConvertHexBytesEx(s_aConvertHexTests[i].pszHex, abBuf, s_aConvertHexTests[i].cbOut,
261 s_aConvertHexTests[i].fColon ? RTSTRCONVERTHEXBYTES_F_SEP_COLON : 0,
262 &pszNext, &cbReturned);
263 if ( rc != s_aConvertHexTests[i].rc
264 || pszNext != pszExpectNext
265 || abBuf[s_aConvertHexTests[i].cbOut - 1] != s_aConvertHexTests[i].bLast
266 )
267 RTTestFailed(hTest, "RTStrConvertHexBytesEx/#%u %s -> %Rrc %p %#zx %#02x, expected %Rrc %p %#zx %#02x\n",
268 i, s_aConvertHexTests[i].pszHex,
269 rc, pszNext, cbReturned, abBuf[s_aConvertHexTests[i].cbOut - 1],
270 s_aConvertHexTests[i].rc, pszExpectNext, s_aConvertHexTests[i].cbOut, s_aConvertHexTests[i].bLast);
271 }
272
273
274 /*
275 * Summary.
276 */
277 return RTTestSummaryAndDestroy(hTest);
278}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette