1ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
2ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*--------------------------------------------------------------------*/
3ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*--- Entirely standalone libc stuff.                 m_libcbase.c ---*/
4ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*--------------------------------------------------------------------*/
5ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
6ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*
7ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   This file is part of Valgrind, a dynamic binary instrumentation
8ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   framework.
9ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
10ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Copyright (C) 2000-2010 Julian Seward
11ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      jseward@acm.org
12ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
13ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   This program is free software; you can redistribute it and/or
14ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   modify it under the terms of the GNU General Public License as
15ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   published by the Free Software Foundation; either version 2 of the
16ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   License, or (at your option) any later version.
17ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
18ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   This program is distributed in the hope that it will be useful, but
19ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   WITHOUT ANY WARRANTY; without even the implied warranty of
20ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
21ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   General Public License for more details.
22ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
23ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   You should have received a copy of the GNU General Public License
24ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   along with this program; if not, write to the Free Software
25ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
26ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   02111-1307, USA.
27ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
28ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   The GNU General Public License is contained in the file COPYING.
29ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown*/
30ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
31ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#include "pub_core_basics.h"
32ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#include "pub_core_libcbase.h"
33ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
34ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* ---------------------------------------------------------------------
35ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char functions.
36ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   ------------------------------------------------------------------ */
37ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
38ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownBool VG_(isspace) ( Char c )
39ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
40ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return (c == ' '  || c == '\n' || c == '\t' ||
41ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown           c == '\f' || c == '\v' || c == '\r');
42ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
43ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
44ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownBool VG_(isdigit) ( Char c )
45ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
46ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return (c >= '0' && c <= '9');
47ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
48ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
49ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* ---------------------------------------------------------------------
50ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Converting strings to numbers
51ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   ------------------------------------------------------------------ */
52ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
53ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownstatic Bool is_dec_digit(Char c, Long* digit)
54ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
55ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (c >= '0' && c <= '9') { *digit = (Long)(c - '0'); return True; }
56ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return False;
57ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
58ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
59ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownstatic Bool is_hex_digit(Char c, Long* digit)
60ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
61ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (c >= '0' && c <= '9') { *digit = (Long)(c - '0');        return True; }
62ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (c >= 'A' && c <= 'F') { *digit = (Long)((c - 'A') + 10); return True; }
63ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (c >= 'a' && c <= 'f') { *digit = (Long)((c - 'a') + 10); return True; }
64ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return False;
65ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
66ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
67ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownLong VG_(strtoll10) ( Char* str, Char** endptr )
68ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
69ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Bool neg = False, converted = False;
70ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Long n = 0, digit = 0;
71ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char* str0 = str;
72ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
73ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Skip leading whitespace.
74ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (VG_(isspace)(*str)) str++;
75ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
76ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Allow a leading '-' or '+'.
77ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (*str == '-') { str++; neg = True; }
78ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   else if (*str == '+') { str++; }
79ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
80ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (is_dec_digit(*str, &digit)) {
81ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      converted = True;          // Ok, we've actually converted a digit.
82ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      n = 10*n + digit;
83ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      str++;
84ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
85ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
86ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (!converted) str = str0;   // If nothing converted, endptr points to
87ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (neg) n = -n;              //   the start of the string.
88ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (endptr) *endptr = str;    // Record first failing character.
89ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return n;
90ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
91ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
92ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownLong VG_(strtoll16) ( Char* str, Char** endptr )
93ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
94ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Bool neg = False, converted = False;
95ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Long n = 0, digit = 0;
96ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char* str0 = str;
97ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
98ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Skip leading whitespace.
99ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (VG_(isspace)(*str)) str++;
100ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
101ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Allow a leading '-' or '+'.
102ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (*str == '-') { str++; neg = True; }
103ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   else if (*str == '+') { str++; }
104ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
105ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Allow leading "0x", but only if there's a hex digit
106ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // following it.
107ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (*str == '0'
108ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown    && (*(str+1) == 'x' || *(str+1) == 'X')
109ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown    && is_hex_digit( *(str+2), &digit )) {
110ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      str += 2;
111ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
112ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
113ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (is_hex_digit(*str, &digit)) {
114ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      converted = True;          // Ok, we've actually converted a digit.
115ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      n = 16*n + digit;
116ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      str++;
117ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
118ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
119ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (!converted) str = str0;   // If nothing converted, endptr points to
120ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (neg) n = -n;              //   the start of the string.
121ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (endptr) *endptr = str;    // Record first failing character.
122ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return n;
123ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
124ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
125ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Browndouble VG_(strtod) ( Char* str, Char** endptr )
126ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
127ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Bool neg = False;
128ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Long digit;
129ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   double n = 0, frac = 0, x = 0.1;
130ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
131ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Skip leading whitespace.
132ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (VG_(isspace)(*str)) str++;
133ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
134ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   // Allow a leading '-' or '+'.
135ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (*str == '-') { str++; neg = True; }
136ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   else if (*str == '+') { str++; }
137ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
138ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (is_dec_digit(*str, &digit)) {
139ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      n = 10*n + digit;
140ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      str++;
141ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
142ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
143ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (*str == '.') {
144ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      str++;
145ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      while (is_dec_digit(*str, &digit)) {
146ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         frac += x*digit;
147ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         x /= 10;
148ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         str++;
149ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
150ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
151ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
152ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   n += frac;
153ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (neg) n = -n;
154ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (endptr) *endptr = str;    // Record first failing character.
155ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return n;
156ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
157ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
158ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar VG_(tolower) ( Char c )
159ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
160ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if ( c >= 'A'  &&  c <= 'Z' ) {
161ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return c - 'A' + 'a';
162ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   } else {
163ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return c;
164ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
165ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
166ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
167ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* ---------------------------------------------------------------------
168ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   String functions
169ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   ------------------------------------------------------------------ */
170ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
171ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownSizeT VG_(strlen) ( const Char* str )
172ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
173ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT i = 0;
174ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (str[i] != 0) i++;
175ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return i;
176ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
177ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
178ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strcat) ( Char* dest, const Char* src )
179ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
180ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char* dest_orig = dest;
181ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*dest) dest++;
182ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*src) *dest++ = *src++;
183ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   *dest = 0;
184ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return dest_orig;
185ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
186ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
187ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strncat) ( Char* dest, const Char* src, SizeT n )
188ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
189ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char* dest_orig = dest;
190ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*dest) dest++;
191ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*src && n > 0) { *dest++ = *src++; n--; }
192ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   *dest = 0;
193ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return dest_orig;
194ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
195ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
196ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strpbrk) ( const Char* s, const Char* accpt )
197ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
198ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   const Char* a;
199ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*s) {
200ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      a = accpt;
201ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      while (*a)
202ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         if (*a++ == *s)
203ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown            return (Char *) s;
204ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s++;
205ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
206ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return NULL;
207ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
208ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
209ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strcpy) ( Char* dest, const Char* src )
210ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
211ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char* dest_orig = dest;
212ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*src) *dest++ = *src++;
213ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   *dest = 0;
214ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return dest_orig;
215ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
216ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
217ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* Copy bytes, not overrunning the end of dest and always ensuring
218ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   zero termination. */
219ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownvoid VG_(strncpy_safely) ( Char* dest, const Char* src, SizeT ndest )
220ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
221ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT i = 0;
222ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
223ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      dest[i] = 0;
224ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (src[i] == 0) return;
225ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (i >= ndest-1) return;
226ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      dest[i] = src[i];
227ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      i++;
228ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
229ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
230ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
231ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strncpy) ( Char* dest, const Char* src, SizeT ndest )
232ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
233ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT i = 0;
234ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
235ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (i >= ndest) return dest;     /* reached limit */
236ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      dest[i] = src[i];
237ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (src[i++] == 0) {
238ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         /* reached NUL;  pad rest with zeroes as required */
239ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         while (i < ndest) dest[i++] = 0;
240ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return dest;
241ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
242ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
243ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
244ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
245ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownInt VG_(strcmp) ( const Char* s1, const Char* s2 )
246ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
247ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
248ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s1 == 0 && *s2 == 0) return 0;
249ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s1 == 0) return -1;
250ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s2 == 0) return 1;
251ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
252ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*(UChar*)s1 < *(UChar*)s2) return -1;
253ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*(UChar*)s1 > *(UChar*)s2) return 1;
254ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
255ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s1++; s2++;
256ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
257ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
258ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
259ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownInt VG_(strcasecmp) ( const Char* s1, const Char* s2 )
260ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
261ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
262ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      UChar c1 = (UChar)VG_(tolower)(*s1);
263ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      UChar c2 = (UChar)VG_(tolower)(*s2);
264ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 == 0 && c2 == 0) return 0;
265ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 == 0) return -1;
266ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c2 == 0) return 1;
267ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
268ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 < c2) return -1;
269ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 > c2) return 1;
270ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
271ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s1++; s2++;
272ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
273ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
274ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
275ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownInt VG_(strncmp) ( const Char* s1, const Char* s2, SizeT nmax )
276ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
277ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT n = 0;
278ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
279ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (n >= nmax) return 0;
280ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s1 == 0 && *s2 == 0) return 0;
281ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s1 == 0) return -1;
282ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s2 == 0) return 1;
283ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
284ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*(UChar*)s1 < *(UChar*)s2) return -1;
285ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*(UChar*)s1 > *(UChar*)s2) return 1;
286ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
287ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s1++; s2++; n++;
288ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
289ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
290ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
291ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownInt VG_(strncasecmp) ( const Char* s1, const Char* s2, SizeT nmax )
292ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
293ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int n = 0;
294ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
295ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      UChar c1;
296ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      UChar c2;
297ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (n >= nmax) return 0;
298ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      c1 = (UChar)VG_(tolower)(*s1);
299ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      c2 = (UChar)VG_(tolower)(*s2);
300ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 == 0 && c2 == 0) return 0;
301ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 == 0) return -1;
302ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c2 == 0) return 1;
303ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
304ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 < c2) return -1;
305ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (c1 > c2) return 1;
306ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
307ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s1++; s2++; n++;
308ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
309ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
310ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
311ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strstr) ( const Char* haystack, Char* needle )
312ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
313ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT n;
314ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (haystack == NULL)
315ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return NULL;
316ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   n = VG_(strlen)(needle);
317ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
318ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (haystack[0] == 0)
319ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return NULL;
320ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (VG_(strncmp)(haystack, needle, n) == 0)
321ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return (Char*)haystack;
322ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      haystack++;
323ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
324ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
325ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
326ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strcasestr) ( const Char* haystack, Char* needle )
327ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
328ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int n;
329ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (haystack == NULL)
330ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return NULL;
331ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   n = VG_(strlen)(needle);
332ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
333ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (haystack[0] == 0)
334ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return NULL;
335ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (VG_(strncasecmp)(haystack, needle, n) == 0)
336ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return (Char*)haystack;
337ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      haystack++;
338ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
339ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
340ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
341ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strchr) ( const Char* s, Char c )
342ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
343ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (True) {
344ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s == c) return (Char*)s;
345ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*s == 0) return NULL;
346ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s++;
347ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
348ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
349ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
350ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownChar* VG_(strrchr) ( const Char* s, Char c )
351ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
352ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int n = VG_(strlen)(s);
353ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (--n > 0) {
354ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (s[n] == c) return (Char*)s + n;
355ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
356ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return NULL;
357ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
358ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
359ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownSizeT VG_(strspn) ( const Char* s, const Char* accpt )
360ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
361ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   const Char *p, *a;
362ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT count = 0;
363ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   for (p = s; *p != '\0'; ++p) {
364ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      for (a = accpt; *a != '\0'; ++a)
365ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         if (*p == *a)
366ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown            break;
367ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (*a == '\0')
368ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return count;
369ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      else
370ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         ++count;
371ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
372ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return count;
373ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
374ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
375ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownSizeT VG_(strcspn) ( const Char* s, const char* reject )
376ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
377ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT count = 0;
378ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (*s != '\0') {
379ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (VG_(strchr) (reject, *s++) == NULL)
380ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         ++count;
381ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      else
382ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return count;
383ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
384ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return count;
385ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
386ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
387ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
388ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* ---------------------------------------------------------------------
389ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   mem* functions
390ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   ------------------------------------------------------------------ */
391ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
392ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownvoid* VG_(memcpy) ( void *dest, const void *src, SizeT sz )
393ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
394ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   const UChar* s  = (const UChar*)src;
395ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         UChar* d  =       (UChar*)dest;
396ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   const UInt*  sI = (const UInt*)src;
397ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         UInt*  dI =       (UInt*)dest;
398ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
399ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (VG_IS_4_ALIGNED(dI) && VG_IS_4_ALIGNED(sI)) {
400ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      while (sz >= 16) {
401ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI[0] = sI[0];
402ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI[1] = sI[1];
403ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI[2] = sI[2];
404ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI[3] = sI[3];
405ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         sz -= 16;
406ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI += 4;
407ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         sI += 4;
408ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
409ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (sz == 0)
410ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return dest;
411ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      while (sz >= 4) {
412ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI[0] = sI[0];
413ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         sz -= 4;
414ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         dI += 1;
415ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         sI += 1;
416ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
417ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (sz == 0)
418ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return dest;
419ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      s = (const UChar*)sI;
420ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d = (UChar*)dI;
421ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
422ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
423ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (sz--)
424ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      *d++ = *s++;
425ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
426ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return dest;
427ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
428ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
429ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownvoid* VG_(memmove)(void *dest, const void *src, SizeT sz)
430ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
431ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT i;
432ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (sz == 0)
433ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return dest;
434ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (dest < src) {
435ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      for (i = 0; i < sz; i++) {
436ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         ((UChar*)dest)[i] = ((UChar*)src)[i];
437ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
438ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
439ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   else if (dest > src) {
440ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      for (i = 0; i < sz; i++) {
441ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         ((UChar*)dest)[sz-i-1] = ((UChar*)src)[sz-i-1];
442ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
443ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
444ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return dest;
445ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
446ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
447ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownvoid* VG_(memset) ( void *destV, Int c, SizeT sz )
448ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
449ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int   c4;
450ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char* d = (Char*)destV;
451ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while ((!VG_IS_4_ALIGNED(d)) && sz >= 1) {
452ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d[0] = c;
453ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d++;
454ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      sz--;
455ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
456ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (sz == 0)
457ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return destV;
458ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   c4 = c & 0xFF;
459ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   c4 |= (c4 << 8);
460ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   c4 |= (c4 << 16);
461ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (sz >= 16) {
462ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      ((Int*)d)[0] = c4;
463ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      ((Int*)d)[1] = c4;
464ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      ((Int*)d)[2] = c4;
465ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      ((Int*)d)[3] = c4;
466ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d += 16;
467ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      sz -= 16;
468ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
469ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (sz >= 4) {
470ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      ((Int*)d)[0] = c4;
471ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d += 4;
472ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      sz -= 4;
473ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
474ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (sz >= 1) {
475ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d[0] = c;
476ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      d++;
477ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      sz--;
478ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
479ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return destV;
480ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
481ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
482ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownInt VG_(memcmp) ( const void* s1, const void* s2, SizeT n )
483ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
484ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int res;
485ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   const UChar *p1 = s1;
486ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   const UChar *p2 = s2;
487ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   UChar a0;
488ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   UChar b0;
489ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
490ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   while (n != 0) {
491ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      a0 = p1[0];
492ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      b0 = p2[0];
493ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      p1 += 1;
494ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      p2 += 1;
495ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      res = a0 - b0;
496ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (res != 0)
497ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         return res;
498ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      n -= 1;
499ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
500ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return 0;
501ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
502ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
503ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* ---------------------------------------------------------------------
504ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Misc useful functions
505ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   ------------------------------------------------------------------ */
506ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
507ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/////////////////////////////////////////////////////////////
508ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/////////////////////////////////////////////////////////////
509ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/// begin Bentley-McIlroy style quicksort
510ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/// See "Engineering a Sort Function".  Jon L Bentley, M. Douglas
511ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/// McIlroy.  Software Practice and Experience Vol 23(11), Nov 1993.
512ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
513ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#define BM_MIN(a, b)                                     \
514ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   (a) < (b) ? a : b
515ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
516ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#define BM_SWAPINIT(a, es)                               \
517ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   swaptype =   ((a-(Char*)0) | es) % sizeof(Word)  ? 2  \
518ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown              : es > (SizeT)sizeof(Word) ? 1             \
519ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown              : 0
520ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
521ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#define BM_EXCH(a, b, t)                                 \
522ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   (t = a, a = b, b = t)
523ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
524ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#define BM_SWAP(a, b)                                    \
525ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   swaptype != 0                                         \
526ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      ? bm_swapfunc(a, b, es, swaptype)                  \
527ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      : (void)BM_EXCH(*(Word*)(a), *(Word*)(b), t)
528ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
529ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#define BM_VECSWAP(a, b, n)                              \
530ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (n > 0) bm_swapfunc(a, b, n, swaptype)
531ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
532ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#define BM_PVINIT(pv, pm)                                \
533ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (swaptype != 0)                                    \
534ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pv = a, BM_SWAP(pv, pm);                           \
535ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   else                                                  \
536ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pv = (Char*)&v, v = *(Word*)pm
537ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
538ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownstatic Char* bm_med3 ( Char* a, Char* b, Char* c,
539ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown                       Int (*cmp)(void*,void*) ) {
540ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return cmp(a, b) < 0
541ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown          ? (cmp(b, c) < 0 ? b : cmp(a, c) < 0 ? c : a)
542ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown          : (cmp(b, c) > 0 ? b : cmp(a, c) > 0 ? c : a);
543ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
544ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
545ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownstatic void bm_swapfunc ( Char* a, Char* b, SizeT n, Int swaptype )
546ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
547ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (swaptype <= 1) {
548ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      Word t;
549ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      for ( ; n > 0; a += sizeof(Word), b += sizeof(Word),
550ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown                                        n -= sizeof(Word))
551ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         BM_EXCH(*(Word*)a, *(Word*)b, t);
552ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   } else {
553ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      Char t;
554ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      for ( ; n > 0; a += 1, b += 1, n -= 1)
555ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         BM_EXCH(*a, *b, t);
556ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
557ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
558ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
559ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownstatic void bm_qsort ( Char* a, SizeT n, SizeT es,
560ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown                       Int (*cmp)(void*,void*) )
561ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
562ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Char  *pa, *pb, *pc, *pd, *pl, *pm, *pn, *pv;
563ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int   r, swaptype;
564ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Word  t, v;
565ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   SizeT s, s1, s2;
566ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown  tailcall:
567ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   BM_SWAPINIT(a, es);
568ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (n < 7) {
569ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      for (pm = a + es; pm < a + n*es; pm += es)
570ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         for (pl = pm; pl > a && cmp(pl-es, pl) > 0; pl -= es)
571ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown            BM_SWAP(pl, pl-es);
572ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      return;
573ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
574ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   pm = a + (n/2)*es;
575ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (n > 7) {
576ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pl = a;
577ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pn = a + (n-1)*es;
578ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (n > 40) {
579ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         s = (n/8)*es;
580ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         pl = bm_med3(pl, pl+s, pl+2*s, cmp);
581ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         pm = bm_med3(pm-s, pm, pm+s, cmp);
582ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         pn = bm_med3(pn-2*s, pn-s, pn, cmp);
583ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
584ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pm = bm_med3(pl, pm, pn, cmp);
585ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
586ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   BM_PVINIT(pv, pm);
587ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   pa = pb = a;
588ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   pc = pd = a + (n-1)*es;
589ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   for (;;) {
590ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      while (pb <= pc && (r = cmp(pb, pv)) <= 0) {
591ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         if (r == 0) { BM_SWAP(pa, pb); pa += es; }
592ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         pb += es;
593ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
594ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      while (pc >= pb && (r = cmp(pc, pv)) >= 0) {
595ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         if (r == 0) { BM_SWAP(pc, pd); pd -= es; }
596ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         pc -= es;
597ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
598ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (pb > pc) break;
599ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      BM_SWAP(pb, pc);
600ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pb += es;
601ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pc -= es;
602ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
603ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   pn = a + n*es;
604ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   s = BM_MIN(pa-a,  pb-pa   ); BM_VECSWAP(a,  pb-s, s);
605ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   s = BM_MIN(pd-pc, pn-pd-es); BM_VECSWAP(pb, pn-s, s);
606ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   /* Now recurse.  Do the smaller partition first with an explicit
607ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      recursion, then do the larger partition using a tail call.
608ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      Except we can't rely on gcc to implement a tail call in any sane
609ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      way, so simply jump back to the start.  This guarantees stack
610ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      growth can never exceed O(log N) even in the worst case. */
611ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   s1 = pb-pa;
612ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   s2 = pd-pc;
613ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (s1 < s2) {
614ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (s1 > es) {
615ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         bm_qsort(a, s1/es, es, cmp);
616ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
617ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (s2 > es) {
618ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         /* bm_qsort(pn-s2, s2/es, es, cmp); */
619ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         a = pn-s2; n = s2/es; es = es; cmp = cmp;
620ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         goto tailcall;
621ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
622ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   } else {
623ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (s2 > es) {
624ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         bm_qsort(pn-s2, s2/es, es, cmp);
625ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
626ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if (s1 > es) {
627ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         /* bm_qsort(a, s1/es, es, cmp); */
628ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         a = a; n = s1/es; es = es; cmp = cmp;
629ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown         goto tailcall;
630ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      }
631ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
632ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
633ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
634ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#undef BM_MIN
635ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#undef BM_SWAPINIT
636ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#undef BM_EXCH
637ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#undef BM_SWAP
638ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#undef BM_VECSWAP
639ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown#undef BM_PVINIT
640ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
641ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/// end Bentley-McIlroy style quicksort
642ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/////////////////////////////////////////////////////////////
643ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/////////////////////////////////////////////////////////////
644ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
645ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/* Returns the base-2 logarithm of x.  Returns -1 if x is not a power
646ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   of two. */
647ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownInt VG_(log2) ( UInt x )
648ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
649ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   Int i;
650ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   /* Any more than 32 and we overflow anyway... */
651ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   for (i = 0; i < 32; i++) {
652ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      if ((1U << i) == x) return i;
653ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   }
654ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return -1;
655ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
656ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
657ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
658ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown// Generic quick sort.
659ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownvoid VG_(ssort)( void* base, SizeT nmemb, SizeT size,
660ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown                 Int (*compar)(void*, void*) )
661ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
662ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   bm_qsort(base,nmemb,size,compar);
663ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
664ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
665ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
666ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown// This random number generator is based on the one suggested in Kernighan
667ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown// and Ritchie's "The C Programming Language".
668ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
669ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown// A pseudo-random number generator returning a random UInt.  If pSeed
670ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown// is NULL, it uses its own seed, which starts at zero.  If pSeed is
671ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown// non-NULL, it uses and updates whatever pSeed points at.
672ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
673ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brownstatic UInt seed = 0;
674ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
675ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff BrownUInt VG_(random)( /*MOD*/UInt* pSeed )
676ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown{
677ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   if (pSeed == NULL)
678ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown      pSeed = &seed;
679ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
680ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   *pSeed = (1103515245 * *pSeed + 12345);
681ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown   return *pSeed;
682ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown}
683ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
684ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*--------------------------------------------------------------------*/
685ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*--- end                                                          ---*/
686ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown/*--------------------------------------------------------------------*/
687ed07e00d438c74b7a23c01bfffde77e3968305e4Jeff Brown
688