5862 lines
140 KiB
C
5862 lines
140 KiB
C
////////////////////////////////////////////////
|
|
////////////////////////////////////////////////
|
|
/////////// BASE IMPLEMENTATION ////////////
|
|
////////////////////////////////////////////////
|
|
////////////////////////////////////////////////
|
|
|
|
////////////////////////////////
|
|
// Functions: Numerical/Math
|
|
|
|
// infinity
|
|
|
|
MR4TH_SYMBOL B32
|
|
is_inf_or_nan_F32(F32 x){
|
|
U32 u = *(U32*)&x;
|
|
B32 result = ((u&0x7f800000) == 0x7f800000);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
is_inf_or_nan_F64(F64 x){
|
|
U64 u = *(U64*)&x;
|
|
B64 result = ((u&0x7ff0000000000000) == 0x7ff0000000000000);
|
|
return(result);
|
|
}
|
|
|
|
// float signs, rounding, and modulus
|
|
|
|
MR4TH_SYMBOL F32
|
|
abs_F32(F32 x){
|
|
union{ F32 f; U32 u; } r;
|
|
r.f = x;
|
|
r.u &= 0x7fffffff;
|
|
return(r.f);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
sign_F32(F32 x){
|
|
union{ F32 f; U32 u; } r;
|
|
r.f = x;
|
|
F32 result = (r.u&0x80000000)?-1.f:1.f;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
trunc_F32(F32 x){
|
|
return((x < 0x1p23)?(F32)(S32)x:x);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
floor_F32(F32 x){
|
|
F32 r = trunc_F32(x);
|
|
if (x < r){
|
|
r -= 1.f;
|
|
}
|
|
return(r);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
ceil_F32(F32 x){
|
|
F32 r = trunc_F32(x);
|
|
if (x > r){
|
|
r += 1.f;
|
|
}
|
|
return(r);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
nearest_F32(F32 x){
|
|
F32 r = floor_F32(x + 0.5f);
|
|
return(r);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
mod_F32(F32 x, F32 m){
|
|
Assert(m > 0.f);
|
|
// + | -
|
|
F32 q = (x/m); // (x/m) | (x/m)
|
|
S32 q_s32 = (S32)q; // floor(x/m) | ceil(x/m)
|
|
F32 r = m*(F32)q_s32;// m*floor(x/m) | m*ceil(x/m)
|
|
// <= x | >= x
|
|
F32 d = x - r;
|
|
if (d < 0){
|
|
d += m;
|
|
}
|
|
return(d);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
frac_F32(F32 x){
|
|
// + | -
|
|
F32 q = (x); // (x) | (x)
|
|
S32 q_s32 = (S32)q; // floor(x) | ceil(x)
|
|
F32 r = (F32)q_s32; // floor(x) | ceil(x)
|
|
// <= x | >= x
|
|
F32 d = x - r;
|
|
return(d);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
abs_F64(F64 x){
|
|
union{ F64 f; U64 u; } r;
|
|
r.f = x;
|
|
r.u &= 0x7fffffffffffffff;
|
|
return(r.f);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
sign_F64(F64 x){
|
|
union{ F64 f; U32 u; } r;
|
|
r.f = x;
|
|
F64 result = (r.u&0x8000000000000000)?-1.:1.;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
trunc_F64(F64 x){
|
|
return((x < 0x1p52)?(F64)(S64)x:x);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
floor_F64(F64 x){
|
|
F64 r = trunc_F64(x);
|
|
if (x < r){
|
|
r -= 1.f;
|
|
}
|
|
return(r);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
ceil_F64(F64 x){
|
|
F64 r = trunc_F64(x);
|
|
if (x > r){
|
|
r += 1.f;
|
|
}
|
|
return(r);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
nearest_F64(F64 x){
|
|
F64 r = floor_F64(x + 0.5f);
|
|
return(r);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
mod_F64(F64 x, F64 m){
|
|
Assert(m > 0.f);
|
|
// + | -
|
|
F64 q = (x/m); // (x/m) | (x/m)
|
|
S64 q_s64 = (S64)q; // floor(x/m) | ceil(x/m)
|
|
F64 r = m*(F64)q_s64;// m*floor(x/m) | m*ceil(x/m)
|
|
// <= x | >= x
|
|
F64 d = x - r;
|
|
if (d < 0){
|
|
d += m;
|
|
}
|
|
return(d);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
frac_F64(F64 x){
|
|
// + | -
|
|
F64 q = (x); // (x) | (x)
|
|
S64 q_s64 = (S64)q; // floor(x) | ceil(x)
|
|
F64 r = (F64)q_s64; // floor(x) | ceil(x)
|
|
// <= x | >= x
|
|
F64 d = x - r;
|
|
return(d);
|
|
}
|
|
|
|
// transcendental functions
|
|
|
|
#include <math.h>
|
|
|
|
MR4TH_SYMBOL F32
|
|
sin_F32(F32 x){
|
|
return(sinf(x*tau_F32));
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
cos_F32(F32 x){
|
|
return(cosf(x*tau_F32));
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
tan_F32(F32 x){
|
|
return(tanf(x*tau_F32));
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
atan_F32(F32 x){
|
|
return(atanf(x)/tau_F32);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
atan2_F32(F32 x, F32 y){
|
|
return(atan2f(y, x)/tau_F32);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
sqrt_F32(F32 x){
|
|
return(sqrtf(x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
inv_sqrt_F32(F32 x){
|
|
return(1.f/sqrtf(x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
ln_F32(F32 x){
|
|
return(logf(x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
pow_F32(F32 base, F32 x){
|
|
return(powf(base, x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
sin_F64(F64 x){
|
|
return(sin(x*tau_F64));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
cos_F64(F64 x){
|
|
return(cos(x*tau_F64));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
tan_F64(F64 x){
|
|
return(tan(x*tau_F64));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
atan_F64(F64 x){
|
|
return(atan(x)/tau_F64);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
atan2_F64(F64 x, F64 y){
|
|
return(atan2(x, y)/tau_F64);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
sqrt_F64(F64 x){
|
|
return(sqrt(x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
inv_sqrt_F64(F64 x){
|
|
return(1.f/sqrt(x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
ln_F64(F64 x){
|
|
return(log(x));
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
pow_F64(F64 base, F64 x){
|
|
return(powf(base, x));
|
|
}
|
|
|
|
// linear interpolation
|
|
|
|
MR4TH_SYMBOL F32
|
|
lerp(F32 a, F32 t, F32 b){
|
|
F32 x = a + (b - a)*t;
|
|
return(x);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
unlerp(F32 a, F32 x, F32 b){
|
|
F32 t = 0.f;
|
|
if (a != b){
|
|
t = (x - a)/(b - a);
|
|
}
|
|
return(t);
|
|
}
|
|
|
|
// integer rounding
|
|
|
|
MR4TH_SYMBOL U32
|
|
next_pow2_U32(U32 x){
|
|
U32 result = x;
|
|
if (result == 0){
|
|
result = 1;
|
|
}
|
|
else{
|
|
// TODO(allen): speed up with bit scan intrinsic
|
|
result -= 1;
|
|
result |= (result >> 1);
|
|
result |= (result >> 2);
|
|
result |= (result >> 4);
|
|
result |= (result >> 8);
|
|
result |= (result >> 16);
|
|
result += 1;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL S32
|
|
sign_extend_S32(S32 x, U32 bitidx){
|
|
U32 shiftn = (31 - bitidx);
|
|
x = x << shiftn;
|
|
x = x >> shiftn;
|
|
return(x);
|
|
}
|
|
|
|
MR4TH_SYMBOL U64
|
|
next_pow2_U64(U32 x){
|
|
U64 result = x;
|
|
if (result == 0){
|
|
result = 1;
|
|
}
|
|
else{
|
|
// TODO(allen): speed up with bit scan intrinsic
|
|
result -= 1;
|
|
result |= (result >> 1);
|
|
result |= (result >> 2);
|
|
result |= (result >> 4);
|
|
result |= (result >> 8);
|
|
result |= (result >> 16);
|
|
result |= (result >> 32);
|
|
result += 1;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL S64
|
|
sign_extend_S64(S64 x, U32 bitidx){
|
|
U32 shiftn = (63 - bitidx);
|
|
x = x << shiftn;
|
|
x = x >> shiftn;
|
|
return(x);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Compound Types
|
|
|
|
// 2d vectors
|
|
|
|
MR4TH_SYMBOL V2F32
|
|
v2_polar(F32 theta, F32 radius){
|
|
V2F32 result = {radius*cos_F32(theta), radius*sin_F32(theta)};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL F32
|
|
angle_from_v2f32(V2F32 v){
|
|
F32 result = atan2_F32(v.x, v.y);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL V2F32
|
|
v2_unit(V2F32 v){
|
|
F32 recip = inv_sqrt_F32(v.x*v.x + v.y*v.y);
|
|
V2F32 r = {0};
|
|
r.x = v.x*recip;
|
|
r.y = v.y*recip;
|
|
return(r);
|
|
}
|
|
|
|
// 3d vectors
|
|
|
|
MR4TH_SYMBOL V3F32
|
|
v3_spherical(F32 theta_xz, F32 theta_yz, F32 radius){
|
|
F32 sxz = sin_F32(theta_xz);
|
|
F32 syz = sin_F32(theta_yz);
|
|
F32 cxz = cos_F32(theta_xz);
|
|
F32 cyz = cos_F32(theta_yz);
|
|
|
|
V3F32 result = {0};
|
|
result.x = radius*sxz;
|
|
result.y = radius*syz;
|
|
result.z = radius*cxz*cyz;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL V3F32
|
|
v3_cross(V3F32 a, V3F32 b){
|
|
V3F32 result = {
|
|
a.y*b.z - a.z*b.y,
|
|
a.z*b.x - a.x*b.z,
|
|
a.x*b.y - a.y*b.x,
|
|
};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL V3F32
|
|
v3_unit(V3F32 v){
|
|
F32 recip = inv_sqrt_F32(v.x*v.x + v.y*v.y + v.z*v.z);
|
|
V3F32 r = {0};
|
|
r.x = v.x*recip;
|
|
r.y = v.y*recip;
|
|
r.z = v.z*recip;
|
|
return(r);
|
|
}
|
|
|
|
// 4x4 matrix
|
|
|
|
MR4TH_SYMBOL B32
|
|
mat4x4_inv(F32 *in, F32 *out){
|
|
/* credit: https://github.com/niswegmann/small-matrix-inverse/tree/master */
|
|
|
|
out[0] =
|
|
+ in[ 5]*in[10]*in[15]
|
|
- in[ 5]*in[11]*in[14]
|
|
- in[ 9]*in[ 6]*in[15]
|
|
+ in[ 9]*in[ 7]*in[14]
|
|
+ in[13]*in[ 6]*in[11]
|
|
- in[13]*in[ 7]*in[10];
|
|
|
|
out[1] =
|
|
- in[ 1]*in[10]*in[15]
|
|
+ in[ 1]*in[11]*in[14]
|
|
+ in[ 9]*in[ 2]*in[15]
|
|
- in[ 9]*in[ 3]*in[14]
|
|
- in[13]*in[ 2]*in[11]
|
|
+ in[13]*in[ 3]*in[10];
|
|
|
|
out[2] =
|
|
+ in[ 1]*in[ 6]*in[15]
|
|
- in[ 1]*in[ 7]*in[14]
|
|
- in[ 5]*in[ 2]*in[15]
|
|
+ in[ 5]*in[ 3]*in[14]
|
|
+ in[13]*in[ 2]*in[ 7]
|
|
- in[13]*in[ 3]*in[ 6];
|
|
|
|
out[3] =
|
|
- in[ 1]*in[ 6]*in[11]
|
|
+ in[ 1]*in[ 7]*in[10]
|
|
+ in[ 5]*in[ 2]*in[11]
|
|
- in[ 5]*in[ 3]*in[10]
|
|
- in[ 9]*in[ 2]*in[ 7]
|
|
+ in[ 9]*in[ 3]*in[ 6];
|
|
|
|
out[4] =
|
|
- in[ 4]*in[10]*in[15]
|
|
+ in[ 4]*in[11]*in[14]
|
|
+ in[ 8]*in[ 6]*in[15]
|
|
- in[ 8]*in[ 7]*in[14]
|
|
- in[12]*in[ 6]*in[11]
|
|
+ in[12]*in[ 7]*in[10];
|
|
|
|
out[5] =
|
|
+ in[ 0]*in[10]*in[15]
|
|
- in[ 0]*in[11]*in[14]
|
|
- in[ 8]*in[ 2]*in[15]
|
|
+ in[ 8]*in[ 3]*in[14]
|
|
+ in[12]*in[ 2]*in[11]
|
|
- in[12]*in[ 3]*in[10];
|
|
|
|
out[6] =
|
|
- in[ 0]*in[ 6]*in[15]
|
|
+ in[ 0]*in[ 7]*in[14]
|
|
+ in[ 4]*in[ 2]*in[15]
|
|
- in[ 4]*in[ 3]*in[14]
|
|
- in[12]*in[ 2]*in[ 7]
|
|
+ in[12]*in[ 3]*in[ 6];
|
|
|
|
out[7] =
|
|
+ in[ 0]*in[ 6]*in[11]
|
|
- in[ 0]*in[ 7]*in[10]
|
|
- in[ 4]*in[ 2]*in[11]
|
|
+ in[ 4]*in[ 3]*in[10]
|
|
+ in[ 8]*in[ 2]*in[ 7]
|
|
- in[ 8]*in[ 3]*in[ 6];
|
|
|
|
out[8] =
|
|
+ in[ 4]*in[ 9]*in[15]
|
|
- in[ 4]*in[11]*in[13]
|
|
- in[ 8]*in[ 5]*in[15]
|
|
+ in[ 8]*in[ 7]*in[13]
|
|
+ in[12]*in[ 5]*in[11]
|
|
- in[12]*in[ 7]*in[ 9];
|
|
|
|
out[9] =
|
|
- in[ 0]*in[ 9]*in[15]
|
|
+ in[ 0]*in[11]*in[13]
|
|
+ in[ 8]*in[ 1]*in[15]
|
|
- in[ 8]*in[ 3]*in[13]
|
|
- in[12]*in[ 1]*in[11]
|
|
+ in[12]*in[ 3]*in[ 9];
|
|
|
|
out[10] =
|
|
+ in[ 0]*in[ 5]*in[15]
|
|
- in[ 0]*in[ 7]*in[13]
|
|
- in[ 4]*in[ 1]*in[15]
|
|
+ in[ 4]*in[ 3]*in[13]
|
|
+ in[12]*in[ 1]*in[ 7]
|
|
- in[12]*in[ 3]*in[ 5];
|
|
|
|
out[11] =
|
|
- in[ 0]*in[ 5]*in[11]
|
|
+ in[ 0]*in[ 7]*in[ 9]
|
|
+ in[ 4]*in[ 1]*in[11]
|
|
- in[ 4]*in[ 3]*in[ 9]
|
|
- in[ 8]*in[ 1]*in[ 7]
|
|
+ in[ 8]*in[ 3]*in[ 5];
|
|
|
|
out[12] =
|
|
- in[ 4]*in[ 9]*in[14]
|
|
+ in[ 4]*in[10]*in[13]
|
|
+ in[ 8]*in[ 5]*in[14]
|
|
- in[ 8]*in[ 6]*in[13]
|
|
- in[12]*in[ 5]*in[10]
|
|
+ in[12]*in[ 6]*in[ 9];
|
|
|
|
out[13] =
|
|
+ in[ 0]*in[ 9]*in[14]
|
|
- in[ 0]*in[10]*in[13]
|
|
- in[ 8]*in[ 1]*in[14]
|
|
+ in[ 8]*in[ 2]*in[13]
|
|
+ in[12]*in[ 1]*in[10]
|
|
- in[12]*in[ 2]*in[ 9];
|
|
|
|
out[14] =
|
|
- in[ 0]*in[ 5]*in[14]
|
|
+ in[ 0]*in[ 6]*in[13]
|
|
+ in[ 4]*in[ 1]*in[14]
|
|
- in[ 4]*in[ 2]*in[13]
|
|
- in[12]*in[ 1]*in[ 6]
|
|
+ in[12]*in[ 2]*in[ 5];
|
|
|
|
out[15] =
|
|
+ in[ 0]*in[ 5]*in[10]
|
|
- in[ 0]*in[ 6]*in[ 9]
|
|
- in[ 4]*in[ 1]*in[10]
|
|
+ in[ 4]*in[ 2]*in[ 9]
|
|
+ in[ 8]*in[ 1]*in[ 6]
|
|
- in[ 8]*in[ 2]*in[ 5];
|
|
|
|
F32 det = in[0]*out[0] + in[1]*out[4] + in[2]*out[8] + in[3]*out[12];
|
|
|
|
B32 result = 0;
|
|
if (det != 0.f){
|
|
F32 invdet = 1.f/det;
|
|
for (U32 i = 0; i < 16; i += 1){
|
|
out[i] *= invdet;
|
|
}
|
|
result = 1;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
mat4x4_mul(F32 *a, F32 *b, F32 *out){
|
|
out[ 0] = a[ 0]*b[ 0] + a[ 1]*b[ 4] + a[ 2]*b[ 8] + a[ 3]*b[12];
|
|
out[ 1] = a[ 0]*b[ 1] + a[ 1]*b[ 5] + a[ 2]*b[ 9] + a[ 3]*b[13];
|
|
out[ 2] = a[ 0]*b[ 2] + a[ 1]*b[ 6] + a[ 2]*b[10] + a[ 3]*b[14];
|
|
out[ 3] = a[ 0]*b[ 3] + a[ 1]*b[ 7] + a[ 2]*b[11] + a[ 3]*b[15];
|
|
out[ 4] = a[ 4]*b[ 0] + a[ 5]*b[ 4] + a[ 6]*b[ 8] + a[ 7]*b[12];
|
|
out[ 5] = a[ 4]*b[ 1] + a[ 5]*b[ 5] + a[ 6]*b[ 9] + a[ 7]*b[13];
|
|
out[ 6] = a[ 4]*b[ 2] + a[ 5]*b[ 6] + a[ 6]*b[10] + a[ 7]*b[14];
|
|
out[ 7] = a[ 4]*b[ 3] + a[ 5]*b[ 7] + a[ 6]*b[11] + a[ 7]*b[15];
|
|
out[ 8] = a[ 8]*b[ 0] + a[ 9]*b[ 4] + a[10]*b[ 8] + a[11]*b[12];
|
|
out[ 9] = a[ 8]*b[ 1] + a[ 9]*b[ 5] + a[10]*b[ 9] + a[11]*b[13];
|
|
out[10] = a[ 8]*b[ 2] + a[ 9]*b[ 6] + a[10]*b[10] + a[11]*b[14];
|
|
out[11] = a[ 8]*b[ 3] + a[ 9]*b[ 7] + a[10]*b[11] + a[11]*b[15];
|
|
out[12] = a[12]*b[ 0] + a[13]*b[ 4] + a[14]*b[ 8] + a[15]*b[12];
|
|
out[13] = a[12]*b[ 1] + a[13]*b[ 5] + a[14]*b[ 9] + a[15]*b[13];
|
|
out[14] = a[12]*b[ 2] + a[13]*b[ 6] + a[14]*b[10] + a[15]*b[14];
|
|
out[15] = a[12]*b[ 3] + a[13]*b[ 7] + a[14]*b[11] + a[15]*b[15];
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Memory Operations
|
|
|
|
MR4TH_SYMBOL void
|
|
memory_zero(void *ptr, U64 size){
|
|
U64 z64 = size/8;
|
|
U64 z8 = size%8;
|
|
|
|
U64 *p64 = (U64*)ptr;
|
|
for (;z64 > 0;){
|
|
*p64 = 0;
|
|
p64 += 1;
|
|
z64 -= 1;
|
|
}
|
|
U8 *p8 = (U8*)p64;
|
|
for (;z8 > 0;){
|
|
*p8 = 0;
|
|
p8 += 1;
|
|
z8 -= 1;
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
memory_fill(void *ptr, U64 size, U8 fillbyte){
|
|
U64 fillqword = fillbyte;
|
|
{
|
|
fillqword |= fillqword << 8;
|
|
fillqword |= fillqword << 16;
|
|
fillqword |= fillqword << 32;
|
|
}
|
|
|
|
U64 z64 = size/8;
|
|
U64 z8 = size%8;
|
|
|
|
U64 *p64 = (U64*)ptr;
|
|
for (;z64 > 0;){
|
|
*p64 = fillqword;
|
|
p64 += 1;
|
|
z64 -= 1;
|
|
}
|
|
U8 *p8 = (U8*)p64;
|
|
for (;z8 > 0;){
|
|
*p8 = fillbyte;
|
|
p8 += 1;
|
|
z8 -= 1;
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
memory_match(void *a, void *b, U64 size){
|
|
U64 z64 = size/8;
|
|
U64 z8 = size%8;
|
|
|
|
U64 *a64 = (U64*)a;
|
|
U64 *b64 = (U64*)b;
|
|
for (;z64 > 0;){
|
|
if (*a64 != *b64){
|
|
return(0);
|
|
}
|
|
a64 += 1;
|
|
b64 += 1;
|
|
z64 -= 1;
|
|
}
|
|
|
|
U8 *a8 = (U8*)a64;
|
|
U8 *b8 = (U8*)b64;
|
|
for (;z8 > 0;){
|
|
if (*a8 != *b8){
|
|
return(0);
|
|
}
|
|
a8 += 1;
|
|
b8 += 1;
|
|
z8 -= 1;
|
|
}
|
|
|
|
return(1);
|
|
}
|
|
|
|
MR4TH_SYMBOL void*
|
|
memory_move(void *dst, void *src, U64 size){
|
|
U64 z64 = size/8;
|
|
U64 z8 = size%8;
|
|
|
|
// backwards
|
|
if ((U8*)src < (U8*)dst){
|
|
U8 *dst8 = ((U8*)dst) + size;
|
|
U8 *src8 = ((U8*)src) + size;
|
|
for (;z8 > 0;){
|
|
dst8 -= 1;
|
|
src8 -= 1;
|
|
*dst8 = *src8;
|
|
z8 -= 1;
|
|
}
|
|
U64 *dst64 = (U64*)dst8;
|
|
U64 *src64 = (U64*)src8;
|
|
for (;z64 > 0;){
|
|
dst64 -= 1;
|
|
src64 -= 1;
|
|
*dst64 = *src64;
|
|
z64 -= 1;
|
|
}
|
|
}
|
|
|
|
// forwards
|
|
else if ((U8*)src > (U8*)dst){
|
|
U64 *dst64 = (U64*)dst;
|
|
U64 *src64 = (U64*)src;
|
|
for (;z64 > 0;){
|
|
*dst64 = *src64;
|
|
dst64 += 1;
|
|
src64 += 1;
|
|
z64 -= 1;
|
|
}
|
|
U8 *dst8 = (U8*)dst64;
|
|
U8 *src8 = (U8*)src64;
|
|
for (;z8 > 0;){
|
|
*dst8 = *src8;
|
|
dst8 += 1;
|
|
src8 += 1;
|
|
z8 -= 1;
|
|
}
|
|
}
|
|
|
|
return(dst);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Symbolic Constants
|
|
|
|
MR4TH_SYMBOL OperatingSystem
|
|
operating_system_from_context(void){
|
|
OperatingSystem result = OperatingSystem_Null;
|
|
#if OS_WINDOWS
|
|
result = OperatingSystem_Windows;
|
|
#elif OS_LINUX
|
|
result = OperatingSystem_Linux;
|
|
#elif OS_MAC
|
|
result = OperatingSystem_Mac;
|
|
#endif
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL Architecture
|
|
architecture_from_context(void){
|
|
Architecture result = Architecture_Null;
|
|
#if ARCH_X64
|
|
result = Architecture_X64;
|
|
#elif ARCH_X86
|
|
result = Architecture_X86;
|
|
#elif ARCH_ARM
|
|
result = Architecture_Arm;
|
|
#elif ARCH_ARM64
|
|
result = Architecture_Arm64;
|
|
#endif
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL char*
|
|
string_from_operating_system(OperatingSystem os){
|
|
char *result = "(null)";
|
|
switch (os){
|
|
default:break;
|
|
case OperatingSystem_Windows:
|
|
{
|
|
result = "windows";
|
|
}break;
|
|
case OperatingSystem_Linux:
|
|
{
|
|
result = "linux";
|
|
}break;
|
|
case OperatingSystem_Mac:
|
|
{
|
|
result = "mac";
|
|
}break;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL char*
|
|
string_from_architecture(Architecture arch){
|
|
char *result = "(null)";
|
|
switch (arch){
|
|
default:break;
|
|
case Architecture_X64:
|
|
{
|
|
result = "x64";
|
|
}break;
|
|
case Architecture_X86:
|
|
{
|
|
result = "x86";
|
|
}break;
|
|
case Architecture_Arm:
|
|
{
|
|
result = "arm";
|
|
}break;
|
|
case Architecture_Arm64:
|
|
{
|
|
result = "armm64";
|
|
}break;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL char*
|
|
string_from_month(Month month){
|
|
char *result = "(null)";
|
|
switch (month){
|
|
case Month_Jan:
|
|
{
|
|
result = "jan";
|
|
}break;
|
|
case Month_Feb:
|
|
{
|
|
result = "feb";
|
|
}break;
|
|
case Month_Mar:
|
|
{
|
|
result = "mar";
|
|
}break;
|
|
case Month_Apr:
|
|
{
|
|
result = "apr";
|
|
}break;
|
|
case Month_May:
|
|
{
|
|
result = "may";
|
|
}break;
|
|
case Month_Jun:
|
|
{
|
|
result = "jun";
|
|
}break;
|
|
case Month_Jul:
|
|
{
|
|
result = "jul";
|
|
}break;
|
|
case Month_Aug:
|
|
{
|
|
result = "aug";
|
|
}break;
|
|
case Month_Sep:
|
|
{
|
|
result = "sep";
|
|
}break;
|
|
case Month_Oct:
|
|
{
|
|
result = "oct";
|
|
}break;
|
|
case Month_Nov:
|
|
{
|
|
result = "nov";
|
|
}break;
|
|
case Month_Dec:
|
|
{
|
|
result = "dec";
|
|
}break;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL char*
|
|
string_from_day_of_week(DayOfWeek day_of_week){
|
|
char *result = "(null)";
|
|
switch (day_of_week){
|
|
case DayOfWeek_Sunday:
|
|
{
|
|
result = "sunday";
|
|
}break;
|
|
case DayOfWeek_Monday:
|
|
{
|
|
result = "monday";
|
|
}break;
|
|
case DayOfWeek_Tuesday:
|
|
{
|
|
result = "tuesday";
|
|
}break;
|
|
case DayOfWeek_Wednesday:
|
|
{
|
|
result = "wednesday";
|
|
}break;
|
|
case DayOfWeek_Thursday:
|
|
{
|
|
result = "thursday";
|
|
}break;
|
|
case DayOfWeek_Friday:
|
|
{
|
|
result = "friday";
|
|
}break;
|
|
case DayOfWeek_Saturday:
|
|
{
|
|
result = "saturday";
|
|
}break;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Time
|
|
|
|
MR4TH_SYMBOL DenseTime
|
|
dense_time_from_date_time(DateTime *in){
|
|
U32 year_encoded = (U32)((S32)in->year + 0x8000);
|
|
DenseTime result = 0;
|
|
result += year_encoded;
|
|
result *= 12;
|
|
result += (in->mon - 1);
|
|
result *= 31;
|
|
result += (in->day - 1);
|
|
result *= 24;
|
|
result += in->hour;
|
|
result *= 60;
|
|
result += in->min;
|
|
result *= 61;
|
|
result += in->sec;
|
|
result *= 1000;
|
|
result += in->msec;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
date_time_from_dense_time(DenseTime in){
|
|
DateTime result = {0};
|
|
result.msec = in%1000;
|
|
in /= 1000;
|
|
result.sec = in%61;
|
|
in /= 61;
|
|
result.min = in%60;
|
|
in /= 60;
|
|
result.hour = in%24;
|
|
in /= 24;
|
|
result.day = (in%31) + 1;
|
|
in /= 31;
|
|
result.mon = (in%12) + 1;
|
|
in /= 12;
|
|
S32 year_encoded = (S32)in;
|
|
result.year = (year_encoded - 0x8000);
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Arena
|
|
|
|
// arena core implementation
|
|
|
|
#define MR4TH_MEM_COMMIT_BLOCK_SIZE MB(64)
|
|
#define MR4TH_MEM_MAX_ALIGN 64
|
|
#define MR4TH_MEM_SCRATCH_POOL_COUNT 2
|
|
|
|
struct Arena{
|
|
Arena *current;
|
|
Arena *prev;
|
|
U64 default_reserve_size;
|
|
U16 alignment;
|
|
B8 growing;
|
|
U8 filler[5];
|
|
U64 base_pos;
|
|
U64 chunk_cap;
|
|
U64 chunk_pos;
|
|
U64 chunk_commit_pos;
|
|
};
|
|
|
|
#define MEM_INITIAL_COMMIT KB(4)
|
|
#define MEM_INTERNAL_MIN_SIZE AlignUpPow2(sizeof(Arena), MR4TH_MEM_MAX_ALIGN)
|
|
|
|
#if !MR4TH_PLUGIN_MODE
|
|
|
|
// arena pre-requisites
|
|
|
|
StaticAssert(sizeof(Arena) <= MEM_INITIAL_COMMIT, mem_check_arena_size);
|
|
|
|
StaticAssert(IsPow2OrZero(MR4TH_MEM_COMMIT_BLOCK_SIZE) &&
|
|
MR4TH_MEM_COMMIT_BLOCK_SIZE != 0,
|
|
mem_check_commit_block_size);
|
|
|
|
StaticAssert(IsPow2OrZero(MR4TH_MEM_MAX_ALIGN) && MR4TH_MEM_MAX_ALIGN != 0,
|
|
mem_check_max_align);
|
|
|
|
// arena functions
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE Arena*
|
|
arena_new(U64 reserve_size, U64 alignment, B32 growing){
|
|
ProfBeginFunc();
|
|
|
|
Arena *result = 0;
|
|
if (reserve_size >= MEM_INITIAL_COMMIT){
|
|
void *memory = os_memory_reserve(reserve_size);
|
|
if (os_memory_commit(memory, MEM_INITIAL_COMMIT)){
|
|
AsanPoison(memory, reserve_size);
|
|
AsanUnpoison(memory, MEM_INTERNAL_MIN_SIZE);
|
|
result = (Arena*)memory;
|
|
result->current = result;
|
|
result->prev = 0;
|
|
result->alignment = alignment;
|
|
result->default_reserve_size = reserve_size;
|
|
result->growing = growing;
|
|
result->base_pos = 0;
|
|
result->chunk_cap = reserve_size;
|
|
result->chunk_pos = MEM_INTERNAL_MIN_SIZE;
|
|
result->chunk_commit_pos = MEM_INITIAL_COMMIT;
|
|
}
|
|
}
|
|
Assert(result != 0);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void
|
|
arena_release(Arena *arena){
|
|
ProfBeginFunc();
|
|
|
|
Arena *ptr = arena->current;
|
|
for (;ptr != 0;){
|
|
Arena *prev = ptr->prev;
|
|
AsanPoison(ptr, ptr->chunk_cap);
|
|
os_memory_release(ptr, ptr->chunk_cap);
|
|
ptr = prev;
|
|
}
|
|
|
|
ProfEndFunc();
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void*
|
|
arena_push_no_zero(Arena *arena, U64 size){
|
|
ProfBeginFunc();
|
|
|
|
void *result = 0;
|
|
Arena *current = arena->current;
|
|
|
|
// allocate new chunk if necessary
|
|
if (arena->growing){
|
|
U64 next_chunk_pos = AlignUpPow2(current->chunk_pos, arena->alignment);
|
|
next_chunk_pos += size;
|
|
if (next_chunk_pos > current->chunk_cap){
|
|
U64 new_reserve_size = arena->default_reserve_size;
|
|
U64 enough_to_fit = size + MEM_INTERNAL_MIN_SIZE;
|
|
if (new_reserve_size < enough_to_fit){
|
|
new_reserve_size = AlignUpPow2(enough_to_fit, KB(4));
|
|
}
|
|
|
|
void *memory = os_memory_reserve(new_reserve_size);
|
|
if (os_memory_commit(memory, MEM_INITIAL_COMMIT)){
|
|
AsanPoison(memory, new_reserve_size);
|
|
AsanUnpoison(memory, MEM_INTERNAL_MIN_SIZE);
|
|
Arena *new_chunk = (Arena*)memory;
|
|
new_chunk->prev = current;
|
|
new_chunk->base_pos = current->base_pos + current->chunk_cap;
|
|
new_chunk->chunk_cap = new_reserve_size;
|
|
new_chunk->chunk_pos = MEM_INTERNAL_MIN_SIZE;
|
|
new_chunk->chunk_commit_pos = MEM_INITIAL_COMMIT;
|
|
current = arena->current = new_chunk;
|
|
}
|
|
}
|
|
}
|
|
|
|
{
|
|
// if there is room in this chunk's reserve ...
|
|
U64 result_pos = AlignUpPow2(current->chunk_pos, arena->alignment);
|
|
U64 next_chunk_pos = result_pos + size;
|
|
if (next_chunk_pos <= current->chunk_cap){
|
|
|
|
// commit more memory if necessary
|
|
if (next_chunk_pos > current->chunk_commit_pos){
|
|
U64 next_commit_pos_aligned =
|
|
AlignUpPow2(next_chunk_pos, MR4TH_MEM_COMMIT_BLOCK_SIZE);
|
|
U64 next_commit_pos =
|
|
ClampTop(next_commit_pos_aligned, current->chunk_cap);
|
|
U64 commit_size = next_commit_pos - current->chunk_commit_pos;
|
|
if (os_memory_commit((U8*)current + current->chunk_commit_pos, commit_size)){
|
|
current->chunk_commit_pos = next_commit_pos;
|
|
}
|
|
}
|
|
|
|
// if there is room in the commit range, return memory & advance pos
|
|
if (next_chunk_pos <= current->chunk_commit_pos){
|
|
AsanUnpoison((U8*)current + current->chunk_pos,
|
|
next_chunk_pos - current->chunk_pos);
|
|
result = (U8*)current + result_pos;
|
|
current->chunk_pos = next_chunk_pos;
|
|
}
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void
|
|
arena_pop_to(Arena *arena, U64 pos){
|
|
ProfBeginFunc();
|
|
|
|
U64 clamped_total_pos = ClampBot(pos, MEM_INTERNAL_MIN_SIZE);
|
|
|
|
Arena *current = arena->current;
|
|
U64 total_pos = current->base_pos + current->chunk_pos;
|
|
if (clamped_total_pos < total_pos){
|
|
// release all chunks that begin after this pos
|
|
for (; clamped_total_pos < current->base_pos; ){
|
|
Arena *prev = current->prev;
|
|
AsanPoison(current, current->chunk_cap);
|
|
os_memory_release(current, current->chunk_cap);
|
|
current = prev;
|
|
}
|
|
|
|
// update arena's current
|
|
{
|
|
arena->current = current;
|
|
}
|
|
|
|
// update the chunk position of the chunk in
|
|
// the chain that contains this position.
|
|
{
|
|
U64 chunk_pos = clamped_total_pos - current->base_pos;
|
|
U64 clamped_chunk_pos = ClampBot(chunk_pos, MEM_INTERNAL_MIN_SIZE);
|
|
AsanPoison((U8*)current + clamped_chunk_pos,
|
|
current->chunk_pos - clamped_chunk_pos);
|
|
current->chunk_pos = clamped_chunk_pos;
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE U64
|
|
arena_current_pos(Arena *arena){
|
|
Arena *current = arena->current;
|
|
U64 result = current->base_pos + current->chunk_pos;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE U64
|
|
arena_current_align(Arena *arena){
|
|
return(arena->alignment);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void
|
|
arena_set_align(Arena *arena, U64 alignment){
|
|
arena->alignment = alignment;
|
|
}
|
|
|
|
MR4TH_SYMBOL_STATIC MR4TH_THREADVAR
|
|
Arena *arena__scratch_pool[MR4TH_MEM_SCRATCH_POOL_COUNT] = {0};
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE ArenaTemp
|
|
arena_get_scratch(Arena **conflict_array, U32 count){
|
|
ProfBeginFunc();
|
|
|
|
// init on first time
|
|
if (arena__scratch_pool[0] == 0){
|
|
Arena **scratch_slot = arena__scratch_pool;
|
|
for (U64 i = 0;
|
|
i < MR4TH_MEM_SCRATCH_POOL_COUNT;
|
|
i += 1, scratch_slot += 1){
|
|
*scratch_slot = arena_alloc();
|
|
}
|
|
}
|
|
|
|
// get non-conflicting arena
|
|
ArenaTemp result = {0};
|
|
Arena **scratch_slot = arena__scratch_pool;
|
|
for (U64 i = 0;
|
|
i < MR4TH_MEM_SCRATCH_POOL_COUNT;
|
|
i += 1, scratch_slot += 1){
|
|
B32 is_non_conflict = 1;
|
|
Arena **conflict_ptr = conflict_array;
|
|
for (U32 j = 0; j < count; j += 1, conflict_ptr += 1){
|
|
if (*scratch_slot == *conflict_ptr){
|
|
is_non_conflict = 0;
|
|
break;
|
|
}
|
|
}
|
|
if (is_non_conflict){
|
|
result = arena_begin_temp(*scratch_slot);
|
|
break;
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
#endif /* arena core implementation */
|
|
|
|
// arena helpers
|
|
|
|
#if !defined(MR4TH_MEM_DEFAULT_RESERVE_SIZE)
|
|
# define MR4TH_MEM_DEFAULT_RESERVE_SIZE GB(1)
|
|
#endif
|
|
#if !defined(MR4TH_MEM_DEFAULT_ALIGNMENT)
|
|
# define MR4TH_MEM_DEFAULT_ALIGNMENT sizeof(void*)
|
|
#endif
|
|
|
|
MR4TH_SYMBOL Arena*
|
|
arena_alloc(void){
|
|
Arena *result = arena_new(MR4TH_MEM_DEFAULT_RESERVE_SIZE, MR4TH_MEM_DEFAULT_ALIGNMENT, 1);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void*
|
|
arena_push(Arena *arena, U64 size){
|
|
void *result = arena_push_no_zero(arena, size);
|
|
MemoryZero(result, size);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
arena_pop_amount(Arena *arena, U64 amount){
|
|
U64 pos = arena_current_pos(arena);
|
|
U64 new_pos = 0;
|
|
if (pos > amount){
|
|
new_pos = pos - amount;
|
|
}
|
|
arena_pop_to(arena, new_pos);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
arena_align(Arena *arena, U64 pow2_align){
|
|
Assert(IsPow2OrZero(pow2_align) && pow2_align != 0 &&
|
|
pow2_align <= MR4TH_MEM_MAX_ALIGN);
|
|
U64 pos = arena_current_pos(arena);
|
|
U64 align_pos = AlignUpPow2(pos, pow2_align);
|
|
U64 z = align_pos - pos;
|
|
if (z > 0){
|
|
arena_push(arena, z);
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL ArenaTemp
|
|
arena_begin_temp(Arena *arena){
|
|
U64 pos = arena_current_pos(arena);
|
|
ArenaTemp temp = {arena, pos};
|
|
return(temp);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
arena_end_temp(ArenaTemp *temp){
|
|
arena_pop_to(temp->arena, temp->pos);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Strings
|
|
|
|
// characters
|
|
|
|
MR4TH_SYMBOL B32
|
|
char_is_whitespace(U8 c){
|
|
return(c == ' ' || c == '\n' || c == '\t' ||
|
|
c == '\r' || c == '\f' || c == '\v');
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
char_is_slash(U8 c){
|
|
return(c == '/' || c == '\\');
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
char_is_digit(U8 c){
|
|
return('0' <= c && c <= '9');
|
|
}
|
|
|
|
MR4TH_SYMBOL U8
|
|
char_to_uppercase(U8 c){
|
|
if ('a' <= c && c <= 'z'){
|
|
c += 'A' - 'a';
|
|
}
|
|
return(c);
|
|
}
|
|
|
|
MR4TH_SYMBOL U8
|
|
char_to_lowercase(U8 c){
|
|
if ('A' <= c && c <= 'Z'){
|
|
c += 'a' - 'A';
|
|
}
|
|
return(c);
|
|
}
|
|
|
|
// in-place constructors
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_range(U8 *first, U8 *opl){
|
|
String8 result = {first, (U64)(opl - first)};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_cstring(U8 *cstr){
|
|
U8 *ptr = cstr;
|
|
for (;*ptr != 0; ptr += 1);
|
|
String8 result = str8_range(cstr, ptr);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_cstring_capped(U8 *cstr, U8 *opl){
|
|
U8 *ptr = cstr;
|
|
for (;ptr < opl && *ptr != 0; ptr += 1);
|
|
String8 result = str8_range(cstr, ptr);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String16
|
|
str16_cstring(U16 *cstr){
|
|
U16 *ptr = cstr;
|
|
for (;*ptr != 0; ptr += 1);
|
|
String16 result = {cstr, (U64)(ptr - cstr)};
|
|
return(result);
|
|
}
|
|
|
|
// compound constructors
|
|
|
|
MR4TH_SYMBOL void
|
|
str8_list_push(Arena *arena, String8List *list, String8 string){
|
|
String8Node *node = push_array(arena, String8Node, 1);
|
|
node->string = string;
|
|
SLLQueuePush(list->first, list->last, node);
|
|
list->node_count += 1;
|
|
list->total_size += string.size;
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
str8_list_push_front(Arena *arena, String8List *list, String8 string){
|
|
String8Node *node = push_array(arena, String8Node, 1);
|
|
node->string = string;
|
|
SLLQueuePushFront(list->first, list->last, node);
|
|
list->node_count += 1;
|
|
list->total_size += string.size;
|
|
}
|
|
|
|
MR4TH_SYMBOL String8List
|
|
str8_list_copy(Arena *arena, String8List *list){
|
|
String8List result = {0};
|
|
for (String8Node *node = list->first;
|
|
node != 0;
|
|
node = node->next){
|
|
String8 string = str8_push_copy(arena, node->string);
|
|
str8_list_push(arena, &result, string);
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_join(Arena *arena, String8List *list, StringJoin *join_optional){
|
|
ProfBeginFunc();
|
|
|
|
// setup join parameters
|
|
MR4TH_SYMBOL_STATIC StringJoin dummy_join = {0};
|
|
StringJoin *join = join_optional;
|
|
if (join == 0){
|
|
join = &dummy_join;
|
|
}
|
|
|
|
// compute total size
|
|
U64 size = (join->pre.size +
|
|
join->post.size +
|
|
((list->node_count>0)?
|
|
(join->mid.size*(list->node_count - 1)):0) +
|
|
list->total_size);
|
|
|
|
// begin string build
|
|
U8 *str = push_array(arena, U8, size + 1);
|
|
U8 *ptr = str;
|
|
|
|
// write pre
|
|
MemoryCopy(ptr, join->pre.str, join->pre.size);
|
|
ptr += join->pre.size;
|
|
|
|
B32 is_mid = 0;
|
|
for (String8Node *node = list->first;
|
|
node != 0;
|
|
node = node->next){
|
|
// write mid
|
|
if (is_mid){
|
|
MemoryCopy(ptr, join->mid.str, join->mid.size);
|
|
ptr += join->mid.size;
|
|
}
|
|
|
|
// write node string
|
|
MemoryCopy(ptr, node->string.str, node->string.size);
|
|
ptr += node->string.size;
|
|
|
|
is_mid = 1;
|
|
}
|
|
|
|
// write post
|
|
MemoryCopy(ptr, join->post.str, join->post.size);
|
|
ptr += join->post.size;
|
|
|
|
// write null
|
|
*ptr = 0;
|
|
|
|
String8 result = CLiteral(String8){str, size};
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8List
|
|
str8_split(Arena *arena, String8 string, U8 *splits, U32 count){
|
|
ProfBeginFunc();
|
|
|
|
String8List result = {0};
|
|
|
|
U8 *ptr = string.str;
|
|
U8 *word_first = ptr;
|
|
U8 *opl = string.str + string.size;
|
|
for (;ptr < opl; ptr += 1){
|
|
// is this a split
|
|
U8 byte = *ptr;
|
|
B32 is_split_byte = 0;
|
|
for (U32 i = 0; i < count; i += 1){
|
|
if (byte == splits[i]){
|
|
is_split_byte = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (is_split_byte){
|
|
// try to emit word, advance word first pointer
|
|
if (word_first < ptr){
|
|
str8_list_push(arena, &result, str8_range(word_first, ptr));
|
|
}
|
|
word_first = ptr + 1;
|
|
}
|
|
}
|
|
|
|
// try to emit final word
|
|
if (word_first < ptr){
|
|
str8_list_push(arena, &result, str8_range(word_first, ptr));
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_pushfv(Arena *arena, char *fmt, va_list args){
|
|
ProfBeginFunc();
|
|
|
|
// in case we need to try a second time
|
|
va_list args2;
|
|
va_copy(args2, args);
|
|
|
|
// try to build the string in 1024 bytes
|
|
U64 buffer_size = 1024;
|
|
U8 *buffer = push_array(arena, U8, buffer_size);
|
|
U64 actual_size = m4_vsnprintf((char*)buffer, buffer_size, fmt, args);
|
|
|
|
String8 result = {0};
|
|
if (actual_size < buffer_size){
|
|
// if first try worked, put back what we didn't use and finish
|
|
arena_pop_amount(arena, buffer_size - actual_size - 1);
|
|
result = CLiteral(String8){buffer, actual_size};
|
|
}
|
|
else{
|
|
// if first try failed, reset and try again with correct size
|
|
arena_pop_amount(arena, buffer_size);
|
|
U8 *fixed_buffer = push_array(arena, U8, actual_size + 1);
|
|
U64 final_size = m4_vsnprintf((char*)fixed_buffer, actual_size + 1, fmt, args2);
|
|
result = CLiteral(String8){fixed_buffer, final_size};
|
|
}
|
|
|
|
// end args2
|
|
va_end(args2);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_pushf(Arena *arena, char *fmt, ...){
|
|
va_list args;
|
|
va_start(args, fmt);
|
|
String8 result = str8_pushfv(arena, fmt, args);
|
|
va_end(args);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
str8_list_pushf(Arena *arena, String8List *list, char *fmt, ...){
|
|
va_list args;
|
|
va_start(args, fmt);
|
|
String8 string = str8_pushfv(arena, fmt, args);
|
|
va_end(args);
|
|
str8_list_push(arena, list, string);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_push_copy(Arena *arena, String8 string){
|
|
String8 result = {0};
|
|
result.str = push_array(arena, U8, string.size + 1);
|
|
result.size = string.size;
|
|
MemoryCopy(result.str, string.str, string.size);
|
|
result.str[result.size] = 0;
|
|
return(result);
|
|
}
|
|
|
|
// substrings
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_prefix(String8 str, U64 size){
|
|
U64 size_clamped = ClampTop(size, str.size);
|
|
String8 result = {str.str, size_clamped};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_chop(String8 str, U64 amount){
|
|
U64 amount_clamped = ClampTop(amount, str.size);
|
|
U64 remaining_size = str.size - amount_clamped;
|
|
String8 result = {str.str, remaining_size};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_postfix(String8 str, U64 size){
|
|
U64 size_clamped = ClampTop(size, str.size);
|
|
U64 skip_to = str.size - size_clamped;
|
|
String8 result = {str.str + skip_to, size_clamped};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_skip(String8 str, U64 amount){
|
|
U64 amount_clamped = ClampTop(amount, str.size);
|
|
U64 remaining_size = str.size - amount_clamped;
|
|
String8 result = {str.str + amount_clamped, remaining_size};
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_skip_chop_whitespace(String8 str){
|
|
String8 result = str;
|
|
if (result.size > 0){
|
|
U8 *sptr = str.str;
|
|
U8 *eptr = str.str + str.size - 1;
|
|
for (;sptr < eptr && char_is_whitespace(*sptr); sptr += 1);
|
|
for (;sptr < eptr && char_is_whitespace(*eptr); eptr -= 1);
|
|
result = str8_range(sptr, eptr + 1);
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
// hash
|
|
|
|
MR4TH_SYMBOL U64
|
|
str8_hash(String8 str){
|
|
U64 hash = 5381;
|
|
for (U8 *ptr = str.str, *opl = str.str + str.size;
|
|
ptr < opl;
|
|
ptr += 1){
|
|
U8 c = *ptr;
|
|
hash = (hash*33)^c;
|
|
}
|
|
return(hash);
|
|
}
|
|
|
|
// compare
|
|
|
|
MR4TH_SYMBOL B32
|
|
str8_match(String8 a, String8 b, StringMatchFlags flags){
|
|
ProfBeginFunc();
|
|
|
|
B32 result = 0;
|
|
if ((flags & StringMatchFlag_PrefixMatch) != 0 || a.size == b.size){
|
|
U64 size = a.size;
|
|
if ((flags & StringMatchFlag_PrefixMatch) != 0){
|
|
size = Min(a.size, b.size);
|
|
}
|
|
|
|
result = 1;
|
|
B32 no_case = ((flags & StringMatchFlag_NoCase) != 0);
|
|
for (U64 i = 0; i < size; i += 1){
|
|
U8 ac = a.str[i];
|
|
U8 bc = b.str[i];
|
|
if (no_case){
|
|
ac = char_to_uppercase(ac);
|
|
bc = char_to_uppercase(bc);
|
|
}
|
|
if (ac != bc){
|
|
result = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
// path helpers
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_chop_last_slash(String8 string){
|
|
String8 result = string;
|
|
if (string.size > 0){
|
|
// pos one past last slash
|
|
U64 pos = string.size;
|
|
for (S64 i = string.size - 1; i >= 0; i -= 1){
|
|
if (char_is_slash(string.str[i])){
|
|
pos = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// chop result string
|
|
result.size = pos;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_file_name_from_path(String8 full_file_name){
|
|
String8 result = {0};
|
|
if (full_file_name.size > 0){
|
|
U8 *opl = full_file_name.str + full_file_name.size;
|
|
U8 *ptr = opl;
|
|
for (;ptr > full_file_name.str; ptr -= 1){
|
|
U8 c = ptr[-1];
|
|
if (c == '/' || c == '\\'){
|
|
break;
|
|
}
|
|
}
|
|
result = str8_range(ptr, opl);
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_base_name_from_file_name(String8 file_name){
|
|
String8 result = {0};
|
|
if (file_name.size > 0){
|
|
U8 *opl = file_name.str + file_name.size;
|
|
U8 *ptr = file_name.str;
|
|
for (;ptr < opl; ptr += 1){
|
|
U8 c = ptr[0];
|
|
if (c == '.'){
|
|
break;
|
|
}
|
|
}
|
|
result = str8_range(file_name.str, ptr);
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
// stylized constructors
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_join_flags(Arena *arena, String8List *list){
|
|
StringJoin join = {0};
|
|
join.mid = str8_lit(" | ");
|
|
String8 result = str8_join(arena, list, &join);
|
|
if (result.size == 0){
|
|
result = str8_lit("0");
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
// unicode
|
|
|
|
MR4TH_SYMBOL StringDecode
|
|
str_decode_utf8(U8 *str, U32 cap){
|
|
ProfBeginFunc();
|
|
|
|
MR4TH_SYMBOL_STATIC U8 length[] = {
|
|
1, 1, 1, 1, // 000xx
|
|
1, 1, 1, 1,
|
|
1, 1, 1, 1,
|
|
1, 1, 1, 1,
|
|
0, 0, 0, 0, // 100xx
|
|
0, 0, 0, 0,
|
|
2, 2, 2, 2, // 110xx
|
|
3, 3, // 1110x
|
|
4, // 11110
|
|
0, // 11111
|
|
};
|
|
MR4TH_SYMBOL_STATIC U8 first_byte_mask[] = { 0, 0x7F, 0x1F, 0x0F, 0x07 };
|
|
MR4TH_SYMBOL_STATIC U8 final_shift[] = { 0, 18, 12, 6, 0 };
|
|
|
|
StringDecode result = {0};
|
|
if (cap > 0){
|
|
result.codepoint = '#';
|
|
result.size = 1;
|
|
|
|
U8 byte = str[0];
|
|
U8 l = length[byte >> 3];
|
|
if (0 < l && l <= cap){
|
|
U32 cp = (byte & first_byte_mask[l]) << 18;
|
|
switch (l){
|
|
case 4: cp |= ((str[3] & 0x3F) << 0);
|
|
case 3: cp |= ((str[2] & 0x3F) << 6);
|
|
case 2: cp |= ((str[1] & 0x3F) << 12);
|
|
default: break;
|
|
}
|
|
cp >>= final_shift[l];
|
|
|
|
result.codepoint = cp;
|
|
result.size = l;
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL U32
|
|
str_encode_utf8(U8 *dst, U32 codepoint){
|
|
ProfBeginFunc();
|
|
|
|
U32 size = 0;
|
|
if (codepoint < (1 << 7)){
|
|
dst[0] = codepoint;
|
|
size = 1;
|
|
}
|
|
else if (codepoint < (1 << 11)){
|
|
dst[0] = 0xC0 | (codepoint >> 6);
|
|
dst[1] = 0x80 | (codepoint & 0x3F);
|
|
size = 2;
|
|
}
|
|
else if (codepoint < (1 << 16)){
|
|
dst[0] = 0xE0 | (codepoint >> 12);
|
|
dst[1] = 0x80 | ((codepoint >> 6) & 0x3F);
|
|
dst[2] = 0x80 | (codepoint & 0x3F);
|
|
size = 3;
|
|
}
|
|
else if (codepoint < (1 << 21)){
|
|
dst[0] = 0xF0 | (codepoint >> 18);
|
|
dst[1] = 0x80 | ((codepoint >> 12) & 0x3F);
|
|
dst[2] = 0x80 | ((codepoint >> 6) & 0x3F);
|
|
dst[3] = 0x80 | (codepoint & 0x3F);
|
|
size = 4;
|
|
}
|
|
else{
|
|
dst[0] = '#';
|
|
size = 1;
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(size);
|
|
}
|
|
|
|
MR4TH_SYMBOL StringDecode
|
|
str_decode_utf16(U16 *str, U32 cap){
|
|
ProfBeginFunc();
|
|
|
|
StringDecode result = {'#', 1};
|
|
U16 x = str[0];
|
|
if (x < 0xD800 || 0xDFFF < x){
|
|
result.codepoint = x;
|
|
}
|
|
else if (cap >= 2){
|
|
U16 y = str[1];
|
|
if (0xD800 <= x && x < 0xDC00 &&
|
|
0xDC00 <= y && y < 0xE000){
|
|
U16 xj = x - 0xD800;
|
|
U16 yj = y - 0xDc00;
|
|
U32 xy = (xj << 10) | yj;
|
|
result.codepoint = xy + 0x10000;
|
|
result.size = 2;
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL U32
|
|
str_encode_utf16(U16 *dst, U32 codepoint){
|
|
ProfBeginFunc();
|
|
|
|
U32 size = 0;
|
|
if (codepoint < 0x10000){
|
|
dst[0] = codepoint;
|
|
size = 1;
|
|
}
|
|
else{
|
|
U32 cpj = codepoint - 0x10000;
|
|
dst[0] = (cpj >> 10) + 0xD800;
|
|
dst[1] = (cpj & 0x3FF) + 0xDC00;
|
|
size = 2;
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(size);
|
|
}
|
|
|
|
MR4TH_SYMBOL String32
|
|
str32_from_str8(Arena *arena, String8 string){
|
|
ProfBeginFunc();
|
|
|
|
U32 *memory = push_array(arena, U32, string.size + 1);
|
|
|
|
U32 *dptr = memory;
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
for (; ptr < opl;){
|
|
StringDecode decode = str_decode_utf8(ptr, (U64)(opl - ptr));
|
|
*dptr = decode.codepoint;
|
|
ptr += decode.size;
|
|
dptr += 1;
|
|
}
|
|
|
|
*dptr = 0;
|
|
|
|
U64 alloc_count = string.size + 1;
|
|
U64 string_count = (U64)(dptr - memory);
|
|
U64 unused_count = alloc_count - string_count - 1;
|
|
arena_pop_amount(arena, unused_count*sizeof(*memory));
|
|
|
|
String32 result = {memory, string_count};
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_from_str32(Arena *arena, String32 string){
|
|
ProfBeginFunc();
|
|
|
|
U8 *memory = push_array(arena, U8, string.size*4 + 1);
|
|
|
|
U8 *dptr = memory;
|
|
U32 *ptr = string.str;
|
|
U32 *opl = string.str + string.size;
|
|
for (; ptr < opl;){
|
|
U32 size = str_encode_utf8(dptr, *ptr);
|
|
ptr += 1;
|
|
dptr += size;
|
|
}
|
|
|
|
*dptr = 0;
|
|
|
|
U64 alloc_count = string.size*4 + 1;
|
|
U64 string_count = (U64)(dptr - memory);
|
|
U64 unused_count = alloc_count - string_count - 1;
|
|
arena_pop_amount(arena, unused_count*sizeof(*memory));
|
|
|
|
String8 result = {memory, string_count};
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String16
|
|
str16_from_str8(Arena *arena, String8 string){
|
|
ProfBeginFunc();
|
|
U16 *memory = push_array(arena, U16, string.size*2 + 1);
|
|
|
|
U16 *dptr = memory;
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
for (; ptr < opl;){
|
|
StringDecode decode = str_decode_utf8(ptr, (U64)(opl - ptr));
|
|
U32 enc_size = str_encode_utf16(dptr, decode.codepoint);
|
|
ptr += decode.size;
|
|
dptr += enc_size;
|
|
}
|
|
|
|
*dptr = 0;
|
|
|
|
U64 alloc_count = string.size*2 + 1;
|
|
U64 string_count = (U64)(dptr - memory);
|
|
U64 unused_count = alloc_count - string_count - 1;
|
|
arena_pop_amount(arena, unused_count*sizeof(*memory));
|
|
|
|
String16 result = {memory, string_count};
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
str8_from_str16(Arena *arena, String16 string){
|
|
ProfBeginFunc();
|
|
U8 *memory = push_array(arena, U8, string.size*3 + 1);
|
|
|
|
U8 *dptr = memory;
|
|
U16 *ptr = string.str;
|
|
U16 *opl = string.str + string.size;
|
|
for (; ptr < opl;){
|
|
StringDecode decode = str_decode_utf16(ptr, (U64)(opl - ptr));
|
|
U16 enc_size = str_encode_utf8(dptr, decode.codepoint);
|
|
ptr += decode.size;
|
|
dptr += enc_size;
|
|
}
|
|
|
|
*dptr = 0;
|
|
|
|
U64 alloc_count = string.size*3 + 1;
|
|
U64 string_count = (U64)(dptr - memory);
|
|
U64 unused_count = alloc_count - string_count - 1;
|
|
arena_pop_amount(arena, unused_count*sizeof(*memory));
|
|
|
|
String8 result = {memory, string_count};
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
// numeric conversion
|
|
|
|
MR4TH_SYMBOL B32
|
|
str8_is_u64(String8 string, U32 radix){
|
|
Assert(2 <= radix && radix <= 16);
|
|
|
|
B32 result = 1;
|
|
{
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
for (; ptr < opl; ptr += 1){
|
|
U8 num = 0xFF;
|
|
|
|
// extract
|
|
U8 s = *ptr - '0';
|
|
if (s <= 9){
|
|
num = s;
|
|
}
|
|
else{
|
|
s = *ptr - 'A';
|
|
if (s <= 5){
|
|
num = 0xA + s;
|
|
}
|
|
else{
|
|
s = *ptr - 'a';
|
|
if (s <= 5){
|
|
num = 0xA + s;
|
|
}
|
|
}
|
|
}
|
|
|
|
// bad parse check
|
|
if (num >= radix){
|
|
result = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL U64
|
|
u64_from_str8(String8 string, U32 radix){
|
|
Assert(2 <= radix && radix <= 16);
|
|
|
|
U64 result = 0;
|
|
{
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
for (; ptr < opl; ptr += 1){
|
|
U8 num = 0xFF;
|
|
|
|
// extract
|
|
U8 s = *ptr - '0';
|
|
if (s <= 9){
|
|
num = s;
|
|
}
|
|
else{
|
|
s = *ptr - 'A';
|
|
if (s <= 5){
|
|
num = 0xA + s;
|
|
}
|
|
else{
|
|
s = *ptr - 'a';
|
|
if (s <= 5){
|
|
num = 0xA + s;
|
|
}
|
|
}
|
|
}
|
|
|
|
// bad parse check
|
|
if (num >= radix){
|
|
result = 0;
|
|
break;
|
|
}
|
|
|
|
// increment result
|
|
result *= radix;
|
|
result += num;
|
|
}
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL U64
|
|
u64_from_str8_c_syntax(String8 string){
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
|
|
U32 radix = 10;
|
|
if (ptr < opl && *ptr == '0'){
|
|
radix = 010;
|
|
ptr += 1;
|
|
if (ptr < opl && *ptr == 'x'){
|
|
radix = 0x10;
|
|
ptr += 1;
|
|
}
|
|
else if (ptr < opl && *ptr == 'b'){
|
|
radix = 2;
|
|
ptr += 1;
|
|
}
|
|
}
|
|
|
|
U64 result = u64_from_str8(str8_range(ptr, opl), radix);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL S64
|
|
s64_from_str8_c_syntax(String8 string){
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
|
|
B32 negative = 0;
|
|
if (ptr < opl){
|
|
if (*ptr == '-'){
|
|
negative = 1;
|
|
ptr += 1;
|
|
}
|
|
else if (*ptr == '+'){
|
|
ptr += 1;
|
|
}
|
|
}
|
|
|
|
S64 result = u64_from_str8_c_syntax(str8_range(ptr, opl));
|
|
if (negative){
|
|
result = -result;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL F64
|
|
f64_from_str8(String8 string){
|
|
F64 result = 0.f;
|
|
|
|
U8 *ptr = string.str;
|
|
U8 *opl = string.str + string.size;
|
|
|
|
B32 negative = 0;
|
|
if (ptr < opl){
|
|
if (*ptr == '-'){
|
|
negative = 1;
|
|
ptr += 1;
|
|
}
|
|
else if (*ptr == '+'){
|
|
ptr += 1;
|
|
}
|
|
}
|
|
|
|
for (; ptr < opl; ptr += 1){
|
|
if (*ptr == '.'){
|
|
ptr += 1;
|
|
break;
|
|
}
|
|
U32 x = 0;
|
|
if ('0' <= *ptr && *ptr <= '9'){
|
|
x = *ptr - '0';
|
|
}
|
|
result *= 10.f;
|
|
result += x;
|
|
}
|
|
|
|
F32 mul = 0.1f;
|
|
for (; ptr < opl; ptr += 1){
|
|
U32 x = 0;
|
|
if ('0' <= *ptr && *ptr <= '9'){
|
|
x = *ptr - '0';
|
|
}
|
|
result += x*mul;
|
|
mul /= 10.f;
|
|
}
|
|
|
|
if (negative){
|
|
result = -result;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Stream
|
|
|
|
// stream core implementation
|
|
|
|
typedef struct STREAM_Node{
|
|
struct STREAM_Node *next;
|
|
U64 size;
|
|
U8 data[0];
|
|
} STREAM_Node;
|
|
|
|
#define STREAM_NODE_FROM_DATA(d) (STREAM_Node*)((U8*)(d) - sizeof(STREAM_Node))
|
|
|
|
struct STREAM{
|
|
Arena *arena;
|
|
U64 clear_pos;
|
|
STREAM_Node *first_node;
|
|
STREAM_Node *last_node;
|
|
U64 buffer_cap;
|
|
U64 total_size;
|
|
STREAM_Node *unused_node;
|
|
};
|
|
|
|
#define STREAM_ALLOC_SIZE MB(16)
|
|
|
|
#if !MR4TH_PLUGIN_MODE
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE STREAM*
|
|
stream_new(void){
|
|
Arena *arena = arena_alloc();
|
|
STREAM *stream = push_array(arena, STREAM, 1);
|
|
stream->arena = arena;
|
|
stream->clear_pos = arena_current_pos(arena);
|
|
return(stream);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void
|
|
stream_release(STREAM *stream){
|
|
stream_clear(stream);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void
|
|
stream_clear(STREAM *stream){
|
|
Arena *arena = stream->arena;
|
|
U64 clear_pos = stream->clear_pos;
|
|
arena_pop_to(arena, clear_pos);
|
|
MemoryZeroStruct(stream);
|
|
stream->arena = arena;
|
|
stream->clear_pos = clear_pos;
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE void
|
|
stream_write(STREAM *stream, String8 data){
|
|
U64 data_pos = 0;
|
|
|
|
for (;data_pos < data.size;){
|
|
U64 remaining_size = data.size - data_pos;
|
|
|
|
// try to get a node with buffer space from the stream
|
|
// (don't return a node with no space!)
|
|
STREAM_Node *node = 0;
|
|
{
|
|
if (stream->buffer_cap > 0){
|
|
node = stream->last_node;
|
|
}
|
|
if (node != 0 && stream->buffer_cap <= node->size){
|
|
node = 0;
|
|
}
|
|
}
|
|
|
|
// if we didn't get a buffer, try the unused node
|
|
if (node == 0){
|
|
if (stream->unused_node != 0){
|
|
node = stream->unused_node;
|
|
stream->unused_node = 0;
|
|
stream->buffer_cap = node->size;
|
|
node->size = 0;
|
|
SLLQueuePush(stream->first_node, stream->last_node, node);
|
|
}
|
|
}
|
|
|
|
// if we didn't get a buffer, make one now
|
|
if (node == 0){
|
|
U64 alloc_size = sizeof(STREAM_Node) + remaining_size;
|
|
alloc_size = ClampBot(STREAM_ALLOC_SIZE, alloc_size);
|
|
|
|
node = (STREAM_Node*)push_array_no_zero(stream->arena, U8, alloc_size);
|
|
SLLQueuePush(stream->first_node, stream->last_node, node);
|
|
node->size = 0;
|
|
|
|
stream->buffer_cap = alloc_size - sizeof(STREAM_Node);
|
|
}
|
|
|
|
// get buffer from the node
|
|
U8 *buffer = node->data + node->size;
|
|
U64 buffer_size = stream->buffer_cap - node->size;
|
|
|
|
// calculate amount to copy
|
|
U64 copy_amount = ClampTop(remaining_size, buffer_size);
|
|
|
|
// copy memory and increment counters
|
|
MemoryCopy(buffer, data.str + data_pos, copy_amount);
|
|
node->size += copy_amount;
|
|
stream->total_size += copy_amount;
|
|
data_pos += copy_amount;
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE U8*
|
|
stream_alloc(STREAM *stream, U64 size){
|
|
// try to get a node with buffer space
|
|
// (don't return a node with no space)
|
|
STREAM_Node *node = 0;
|
|
{
|
|
if (stream->buffer_cap > 0){
|
|
node = stream->last_node;
|
|
}
|
|
if (node != 0 && stream->buffer_cap <= node->size){
|
|
node = 0;
|
|
}
|
|
}
|
|
|
|
// if there is a node but it's not big enough...
|
|
if (node != 0 && stream->buffer_cap - node->size < size){
|
|
U64 remaining_cap = stream->buffer_cap - node->size;
|
|
U64 unused_ptr = IntFromPtr(node->data) + node->size;
|
|
U64 unused_opl = IntFromPtr(node->data) + stream->buffer_cap;
|
|
U64 unused_ptr_aligned = AlignUpPow2(unused_ptr, sizeof(void*));
|
|
if (unused_opl > unused_ptr_aligned &&
|
|
unused_opl - unused_ptr_aligned >= sizeof(STREAM_Node) + 32){
|
|
STREAM_Node *unused_node = (STREAM_Node*)PtrFromInt(unused_ptr_aligned);
|
|
U64 leftover_buffer_size = unused_opl - unused_ptr_aligned - sizeof(STREAM_Node);
|
|
unused_node->size = leftover_buffer_size;
|
|
SLLStackPush(stream->unused_node, unused_node);
|
|
}
|
|
node = 0;
|
|
}
|
|
|
|
// if we don't have a node then setup a new node with sufficient space.
|
|
if (node == 0){
|
|
U64 alloc_size = sizeof(STREAM_Node) + size;
|
|
alloc_size = ClampBot(STREAM_ALLOC_SIZE, alloc_size);
|
|
|
|
node = (STREAM_Node*)push_array_no_zero(stream->arena, U8, alloc_size);
|
|
SLLQueuePush(stream->first_node, stream->last_node, node);
|
|
node->size = 0;
|
|
|
|
stream->buffer_cap = alloc_size - sizeof(STREAM_Node);
|
|
}
|
|
|
|
// return the buffer
|
|
U8 *result = node->data + node->size;
|
|
node->size += size;
|
|
stream->total_size += size;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE U64
|
|
stream_total_size(STREAM *stream){
|
|
return(stream->total_size);
|
|
}
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE STREAM_Handle*
|
|
stream_next_chunk(STREAM *stream, STREAM_Handle *handle, String8 *chunk_out){
|
|
STREAM_Node *node = (STREAM_Node*)handle;
|
|
if (node == 0){
|
|
node = stream->first_node;
|
|
}
|
|
else{
|
|
node = node->next;
|
|
}
|
|
if (node != 0){
|
|
chunk_out->size = node->size;
|
|
chunk_out->str = node->data;
|
|
}
|
|
else{
|
|
chunk_out->size = 0;
|
|
chunk_out->str = 0;
|
|
}
|
|
return((STREAM_Handle*)node);
|
|
}
|
|
|
|
#endif /* stream core implementation */
|
|
|
|
// stream helpers
|
|
|
|
MR4TH_SYMBOL String8
|
|
stream_read(Arena *arena, STREAM *stream){
|
|
String8 result = {0};
|
|
result.str = push_array(arena, U8, stream->total_size);
|
|
result.size = stream->total_size;
|
|
|
|
U64 pos = 0;
|
|
for (STREAM_Node *node = stream->first_node;
|
|
node != 0;
|
|
node = node->next){
|
|
MemoryCopy(result.str + pos, node->data, node->size);
|
|
pos += node->size;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
stream_printfv(STREAM *stream, char *fmt, va_list args){
|
|
// in case we need to try a second time
|
|
va_list args2;
|
|
va_copy(args2, args);
|
|
|
|
// try to get a buffer from the stream
|
|
STREAM_Node *node = 0;
|
|
U8 *buffer = 0;
|
|
U64 buffer_size = 0;
|
|
if (stream->buffer_cap > 0){
|
|
node = stream->last_node;
|
|
buffer = node->data + node->size;
|
|
buffer_size = stream->buffer_cap - node->size;
|
|
}
|
|
|
|
// try to build the string in the current block
|
|
U64 actual_size = 0;
|
|
if (buffer_size > 0){
|
|
actual_size = m4_vsnprintf((char*)buffer, buffer_size, fmt, args);
|
|
}
|
|
else{
|
|
actual_size = m4_vsnprintf(0, 0, fmt, args);
|
|
}
|
|
|
|
// if first try worked, increment block size
|
|
if (actual_size <= buffer_size){
|
|
node->size += actual_size;
|
|
stream->total_size += actual_size;
|
|
}
|
|
|
|
// if first try failed, reset and try again with correct size
|
|
else{
|
|
// TODO(allen): super annoying extra buffer copy, need to fix.
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
U8 *temp_buffer = push_array(scratch.arena, U8,actual_size + 1);
|
|
U64 final_size = m4_vsnprintf((char*)temp_buffer, actual_size + 1, fmt, args2);
|
|
Assert(final_size == actual_size);
|
|
U8 *new_buffer = stream_alloc(stream, actual_size);
|
|
MemoryCopy(new_buffer, temp_buffer, actual_size);
|
|
arena_release_scratch(&scratch);
|
|
}
|
|
|
|
// end args2
|
|
va_end(args2);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
stream_printf(STREAM *stream, char *fmt, ...){
|
|
va_list args;
|
|
va_start(args, fmt);
|
|
stream_printfv(stream, fmt, args);
|
|
va_end(args);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Buffer
|
|
|
|
MR4TH_SYMBOL void
|
|
buffer_alloc(BUFFER *buffer, U64 size){
|
|
MemoryZeroStruct(buffer);
|
|
U64 real_size = AlignUpPow2(size, KB(4));
|
|
buffer->base = os_memory_reserve(real_size);
|
|
if (buffer->base != 0){
|
|
buffer->size = real_size;
|
|
buffer->commit_pos = 0;
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
buffer_release(BUFFER *buffer){
|
|
os_memory_release(buffer->base, buffer->size);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
buffer_reset(BUFFER *buffer){
|
|
if (buffer->commit_pos > 0){
|
|
os_memory_decommit((U8*)buffer->base, buffer->commit_pos);
|
|
}
|
|
buffer->commit_pos = 0;
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
buffer_commit_off__call(BUFFER *buffer, U64 off){
|
|
U64 new_commit_pos = AlignUpPow2(off, MR4TH_MEM_COMMIT_BLOCK_SIZE);
|
|
new_commit_pos = ClampTop(new_commit_pos, buffer->size);
|
|
if (os_memory_commit((U8*)buffer->base + buffer->commit_pos, new_commit_pos - buffer->commit_pos)){
|
|
buffer->commit_pos = new_commit_pos;
|
|
}
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Types: Hash Table
|
|
|
|
MR4TH_SYMBOL void
|
|
hash_buckets_init(BUFFER *buckets, U64 count){
|
|
Assert(count > 0 && IsPow2OrZero(count));
|
|
buffer_commit_off(buckets, sizeof(void*)*count);
|
|
}
|
|
|
|
MR4TH_SYMBOL HASH_Node*
|
|
hash_buckets_first(BUFFER *buckets, U64 hash){
|
|
U64 count = buckets->commit_pos/sizeof(void*);
|
|
U64 bidx = hash&(count - 1);
|
|
HASH_Node *result = ((HASH_Node**)buckets->base)[bidx];
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
hash_buckets_expand(BUFFER *buckets, U64 capacity){
|
|
static const U32 ALPHA_NUMERATOR = 7;
|
|
static const U32 ALPHA_DENOMINATOR = 8;
|
|
|
|
U64 count = buckets->commit_pos/sizeof(void*);
|
|
U64 max = buckets->size/sizeof(void*);
|
|
U32 dbl_count = count*2;
|
|
if (capacity*ALPHA_DENOMINATOR > count*ALPHA_NUMERATOR &&
|
|
dbl_count <= max){
|
|
U32 old_count = count;
|
|
U32 new_count = count*4;
|
|
if (new_count > max){
|
|
new_count = dbl_count;
|
|
}
|
|
buffer_commit_off(buckets, sizeof(void*)*new_count);
|
|
U64 mask = new_count - 1;
|
|
U64 rehash_mask = mask&~(old_count - 1);
|
|
HASH_Node **buckets_base = (HASH_Node**)buckets->base;
|
|
for (U32 i = 0; i < old_count; i += 1){
|
|
for (HASH_Node **ptr = buckets_base + i;
|
|
*ptr != 0;){
|
|
HASH_Node *node = *ptr;
|
|
if ((node->hash&rehash_mask) != 0){
|
|
U64 bidx = node->hash&mask;
|
|
*ptr = (*ptr)->next;
|
|
SLLStackPush(buckets_base[bidx], node);
|
|
}
|
|
else{
|
|
ptr = &node->next;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
hash_buckets_insert(Arena *arena, BUFFER *buckets, U64 hash, U64 val){
|
|
U64 count = buckets->commit_pos/sizeof(void*);
|
|
U64 bucket_idx = hash&(count - 1);
|
|
HASH_Node *bucket = push_array(arena, HASH_Node, 1);
|
|
bucket->hash = hash;
|
|
bucket->val = val;
|
|
SLLStackPush(((HASH_Node**)buckets->base)[bucket_idx], bucket);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Log
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE MR4TH_THREADVAR
|
|
LOG_ThreadVars *log_vars = 0;
|
|
|
|
MR4TH_SYMBOL void
|
|
log_accum_begin(LOG_LogToProc *proc, void *uptr){
|
|
LOG_ThreadVars *vars = log_vars;
|
|
if (vars == 0){
|
|
Arena *arena = arena_alloc();
|
|
vars = log_vars = push_array(arena, LOG_ThreadVars, 1);
|
|
vars->arena = arena;
|
|
}
|
|
|
|
U64 pos = arena_current_pos(vars->arena);
|
|
LOG_Node *node = push_array(vars->arena, LOG_Node, 1);
|
|
node->pos = pos;
|
|
node->logto = proc;
|
|
node->uptr = uptr;
|
|
SLLStackPush(vars->stack, node);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
log_gathering(void){
|
|
LOG_ThreadVars *vars = log_vars;
|
|
return(vars != 0 && vars->stack != 0);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
log_emit(String8 message){
|
|
LOG_ThreadVars *vars = log_vars;
|
|
if (vars != 0 && vars->stack != 0){
|
|
LOG_Node *node = vars->stack;
|
|
String8 msg_copy = str8_push_copy(vars->arena, message);
|
|
str8_list_push(vars->arena, &node->log, msg_copy);
|
|
if (node->logto != 0){
|
|
node->logto(node->uptr, msg_copy);
|
|
}
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
log_emitf(char *fmt, ...){
|
|
LOG_ThreadVars *vars = log_vars;
|
|
if (vars != 0 && vars->stack != 0){
|
|
LOG_Node *node = vars->stack;
|
|
va_list args;
|
|
va_start(args, fmt);
|
|
String8 string = str8_pushfv(vars->arena, fmt, args);
|
|
va_end(args);
|
|
str8_list_push(vars->arena, &node->log, string);
|
|
if (node->logto != 0){
|
|
node->logto(node->uptr, string);
|
|
}
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
log_accum_end(Arena *arena){
|
|
String8 result = {0};
|
|
LOG_ThreadVars *vars = log_vars;
|
|
if (vars != 0){
|
|
LOG_Node *node = vars->stack;
|
|
if (node != 0){
|
|
result = str8_join(arena, &node->log, 0);
|
|
SLLStackPop(vars->stack);
|
|
arena_pop_to(vars->arena, node->pos);
|
|
}
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Functions: Errors
|
|
|
|
// IMPORTANT: It is important that strings get pushed
|
|
// *on top* of the node that holds them within the arena.
|
|
// basically: Assert((U8*)node < (U8*)node->error.str);
|
|
// except that stops working if the arena is chained...
|
|
|
|
MR4TH_SYMBOL_MUST_SHARE MR4TH_THREADVAR
|
|
ER_ThreadVars *er_vars = 0;
|
|
|
|
MR4TH_SYMBOL void
|
|
er_accum_begin(void){
|
|
ER_ThreadVars *vars = er_vars;
|
|
if (vars == 0){
|
|
Arena *arena = arena_new(KB(64), 1, 0);
|
|
vars = er_vars = push_array(arena, ER_ThreadVars, 1);
|
|
vars->arena = arena;
|
|
}
|
|
|
|
U64 pos = arena_current_pos(vars->arena);
|
|
ER_Node *node = push_array(vars->arena, ER_Node, 1);
|
|
if (node == 0){
|
|
vars->over_stack += 1;
|
|
}
|
|
else{
|
|
node->pos = pos;
|
|
SLLStackPush(vars->stack, node);
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
er_emit(String8 error){
|
|
ER_ThreadVars *vars = er_vars;
|
|
if (vars != 0 &&
|
|
vars->over_stack == 0){
|
|
ER_Node *node = vars->stack;
|
|
if (node != 0 && node->error.size == 0){
|
|
node->error = str8_push_copy(vars->arena, error);
|
|
}
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
er_emitf(char *fmt, ...){
|
|
ER_ThreadVars *vars = er_vars;
|
|
if (vars != 0 &&
|
|
vars->over_stack == 0){
|
|
ER_Node *node = vars->stack;
|
|
if (node != 0 && node->error.size == 0){
|
|
va_list args;
|
|
va_start(args, fmt);
|
|
String8 string = str8_pushfv(vars->arena, fmt, args);
|
|
va_end(args);
|
|
node->error = string;
|
|
}
|
|
}
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
er_accum_end(Arena *arena){
|
|
String8 result = {0};
|
|
ER_ThreadVars *vars = er_vars;
|
|
if (vars != 0){
|
|
if (vars->over_stack == 0){
|
|
ER_Node *node = vars->stack;
|
|
if (node != 0){
|
|
result = str8_push_copy(arena, node->error);
|
|
SLLStackPop(vars->stack);
|
|
arena_pop_to(vars->arena, node->pos);
|
|
}
|
|
}
|
|
else{
|
|
vars->over_stack -= 1;
|
|
}
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////
|
|
////////////////////////////////////////////////
|
|
//////// OS IMPLEMENTATION SHARED //////////
|
|
////////////////////////////////////////////////
|
|
////////////////////////////////////////////////
|
|
|
|
////////////////////////////////
|
|
// Helper Implementation: File Handling
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_write(String8 file_name, String8 data){
|
|
String8Node node = {0};
|
|
node.string = data;
|
|
B32 result = os_file_write_list(file_name, &node);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
os__file_write_callback__list(void *udata){
|
|
String8Node **ptr = (String8Node**)udata;
|
|
String8 result = {0};
|
|
if (*ptr != 0){
|
|
result = (*ptr)->string;
|
|
(*ptr) = (*ptr)->next;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_write_list(String8 file_name, String8Node *first_node){
|
|
B32 result = os_file_write_callback(file_name, &first_node, os__file_write_callback__list);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
os__file_write_callback__stream(void *udata){
|
|
STREAM *stream = *(STREAM**) ((void**)udata + 0);
|
|
STREAM_Handle**handle = (STREAM_Handle**)((void**)udata + 1);
|
|
String8 result = {0};
|
|
*handle = stream_next_chunk(stream, *handle, &result);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_write_stream(String8 file_name, STREAM *stream){
|
|
void *udata[2] = { stream, 0 };
|
|
B32 result = os_file_write_callback(file_name, udata, os__file_write_callback__stream);
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Helper Implementation: Libraries
|
|
|
|
MR4TH_SYMBOL OS_VRange
|
|
os_this_image(void){
|
|
OS_VRange result = {0};
|
|
OS_Library *this_lib = os_lib_from_addr((void*)os_this_image);
|
|
if (this_lib != 0){
|
|
result = os_lib_image_range(this_lib);
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
|
|
|
|
/*******************************
|
|
** Begin Base Win32 **
|
|
*******************************/
|
|
|
|
#if OS_WINDOWS
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: Global Variables
|
|
|
|
MR4TH_SYMBOL_STATIC U64 w32_ticks_per_second = 1;
|
|
|
|
MR4TH_SYMBOL_STATIC Arena *w32_perm_arena = 0;
|
|
MR4TH_SYMBOL_STATIC String8 w32_binary_path = {0};
|
|
MR4TH_SYMBOL_STATIC String8 w32_user_path = {0};
|
|
MR4TH_SYMBOL_STATIC String8 w32_temp_path = {0};
|
|
|
|
MR4TH_SYMBOL_STATIC String8List w32_cmd_line = {0};
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: Process Setup
|
|
|
|
MR4TH_SYMBOL void
|
|
os_main_init(int argc, char **argv){
|
|
ProfBeginFunc();
|
|
|
|
// setup precision time
|
|
LARGE_INTEGER perf_freq = {0};
|
|
if (QueryPerformanceFrequency(&perf_freq)){
|
|
w32_ticks_per_second = ((U64)perf_freq.HighPart << 32) | perf_freq.LowPart;
|
|
}
|
|
timeBeginPeriod(1);
|
|
|
|
// arena
|
|
w32_perm_arena = arena_alloc();
|
|
|
|
// command line arguments
|
|
for (int i = 0; i < argc; i += 1){
|
|
String8 arg = str8_cstring((U8*)argv[i]);
|
|
str8_list_push(w32_perm_arena, &w32_cmd_line, arg);
|
|
}
|
|
|
|
// paths
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
|
|
// binary path
|
|
{
|
|
DWORD cap = 2048;
|
|
U16 *buffer = 0;
|
|
DWORD size = 0;
|
|
for (U64 r = 0; r < 4; r += 1, cap *= 4){
|
|
U16 *try_buffer = push_array(scratch.arena, U16, cap);
|
|
DWORD try_size = GetModuleFileNameW(0, (WCHAR*)try_buffer, cap);
|
|
if (try_size == cap && GetLastError() == ERROR_INSUFFICIENT_BUFFER){
|
|
arena_end_temp(&scratch);
|
|
}
|
|
else{
|
|
buffer = try_buffer;
|
|
size = try_size;
|
|
break;
|
|
}
|
|
}
|
|
|
|
String8 full_path = str8_from_str16(scratch.arena, CLiteral(String16){buffer, size});
|
|
String8 binary_path = str8_chop_last_slash(full_path);
|
|
w32_binary_path = str8_push_copy(w32_perm_arena, binary_path);
|
|
}
|
|
|
|
// user data
|
|
{
|
|
HANDLE token = GetCurrentProcessToken();
|
|
DWORD cap = 2048;
|
|
U16 *buffer = push_array(scratch.arena, U16, cap);
|
|
if (!GetUserProfileDirectoryW(token, (WCHAR*)buffer, &cap)){
|
|
arena_end_temp(&scratch);
|
|
buffer = push_array(scratch.arena, U16, cap + 1);
|
|
if (GetUserProfileDirectoryW(token, (WCHAR*)buffer, &cap)){
|
|
buffer = 0;
|
|
}
|
|
}
|
|
|
|
if (buffer != 0){
|
|
// the docs make it sound like we can only count on
|
|
// cap getting the size on failure; so we're just going to cstring
|
|
// this to be safe.
|
|
w32_user_path = str8_from_str16(w32_perm_arena, str16_cstring(buffer));
|
|
}
|
|
}
|
|
|
|
// temp data
|
|
{
|
|
DWORD cap = 2048;
|
|
U16 *buffer = push_array(scratch.arena, U16, cap);
|
|
DWORD size = GetTempPathW(cap, (WCHAR*)buffer);
|
|
if (size >= cap){
|
|
arena_end_temp(&scratch);
|
|
buffer = push_array(scratch.arena, U16, size + 1);
|
|
size = GetTempPathW(size + 1, (WCHAR*)buffer);
|
|
}
|
|
|
|
// size - 1, because this particular string function
|
|
// in the Win32 API is different from the others and it includes
|
|
// the trailing backslash. We want consistency, so the "- 1" removes it.
|
|
w32_temp_path = str8_from_str16(w32_perm_arena, CLiteral(String16){buffer, size - 1});
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
}
|
|
|
|
MR4TH_SYMBOL String8List
|
|
os_command_line_arguments(void){
|
|
String8List result = w32_cmd_line;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_exit_process(U32 code){
|
|
ProfClose();
|
|
ExitProcess(code);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: Memory Functions
|
|
|
|
MR4TH_SYMBOL void*
|
|
os_memory_reserve(U64 size){
|
|
void *result = VirtualAlloc(0, size, MEM_RESERVE, PAGE_READWRITE);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_memory_commit(void *ptr, U64 size){
|
|
B32 result = (VirtualAlloc(ptr, size, MEM_COMMIT, PAGE_READWRITE) != 0);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_memory_decommit(void *ptr, U64 size){
|
|
VirtualFree(ptr, size, MEM_DECOMMIT);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_memory_release(void *ptr, U64 size){
|
|
VirtualFree(ptr, 0, MEM_RELEASE);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_memory_protect(void *ptr, U64 size, DataAccessFlags access_flags){
|
|
DWORD protect = PAGE_NOACCESS;
|
|
switch (access_flags&0x7){
|
|
case 0: protect = PAGE_NOACCESS; break;
|
|
case DataAccessFlag_Read: protect = PAGE_READONLY; break;
|
|
case DataAccessFlag_Write:
|
|
case DataAccessFlag_Read|DataAccessFlag_Write: protect = PAGE_READWRITE; break;
|
|
case DataAccessFlag_Execute: protect = PAGE_EXECUTE; break;
|
|
case DataAccessFlag_Read|DataAccessFlag_Execute: protect = PAGE_EXECUTE_READ; break;
|
|
case DataAccessFlag_Write|DataAccessFlag_Execute:
|
|
case DataAccessFlag_Read|DataAccessFlag_Write|DataAccessFlag_Execute: protect = PAGE_EXECUTE_READWRITE; break;
|
|
}
|
|
|
|
DWORD dummy = 0;
|
|
B32 result = VirtualProtect(ptr, size, protect, &dummy);
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: File Handling
|
|
|
|
MR4TH_SYMBOL String8
|
|
os_file_read(Arena *arena, String8 file_name){
|
|
ProfBeginFunc();
|
|
|
|
// get handle
|
|
ArenaTemp scratch = arena_get_scratch(&arena, 1);
|
|
String16 file_name16 = str16_from_str8(scratch.arena, file_name);
|
|
HANDLE file = CreateFileW((WCHAR*)file_name16.str,
|
|
GENERIC_READ, 0, 0,
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL,
|
|
0);
|
|
|
|
String8 result = {0};
|
|
if (file != INVALID_HANDLE_VALUE){
|
|
// get size
|
|
DWORD hi_size = 0;
|
|
DWORD lo_size = GetFileSize(file, &hi_size);
|
|
U64 total_size = (((U64)hi_size) << 32) | (U64)lo_size;
|
|
|
|
// allocate buffer
|
|
ArenaTemp restore_point = arena_begin_temp(arena);
|
|
U8 *buffer = push_array_no_zero(arena, U8, total_size + 1);
|
|
|
|
// read
|
|
U8 *ptr = buffer;
|
|
U8 *opl = buffer + total_size;
|
|
B32 success = 1;
|
|
for (;ptr < opl;){
|
|
U64 total_to_read = (U64)(opl - ptr);
|
|
DWORD to_read = (DWORD)total_to_read;
|
|
if (total_to_read > max_U32){
|
|
to_read = max_U32;
|
|
}
|
|
DWORD actual_read = 0;
|
|
if (!ReadFile(file, ptr, to_read, &actual_read, 0)){
|
|
success = 0;
|
|
break;
|
|
}
|
|
ptr += actual_read;
|
|
}
|
|
|
|
// set result or reset memory
|
|
if (success){
|
|
buffer[total_size] = 0;
|
|
result.str = buffer;
|
|
result.size = total_size;
|
|
}
|
|
else{
|
|
arena_end_temp(&restore_point);
|
|
}
|
|
|
|
CloseHandle(file);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_write_callback(String8 file_name, void *udata, OS_FileWriteCallback *callback){
|
|
ProfBeginFunc();
|
|
|
|
// get handle
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 file_name16 = str16_from_str8(scratch.arena, file_name);
|
|
HANDLE file = CreateFileW((WCHAR*)file_name16.str,
|
|
GENERIC_WRITE, 0, 0,
|
|
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
|
|
0);
|
|
|
|
B32 result = 0;
|
|
if (file != INVALID_HANDLE_VALUE){
|
|
result = 1;
|
|
|
|
for (;;){
|
|
String8 chunk = callback(udata);
|
|
if (chunk.size == 0){
|
|
break;
|
|
}
|
|
|
|
U8 *ptr = chunk.str;
|
|
U8 *opl = chunk.str + chunk.size;
|
|
for (;ptr < opl;){
|
|
U64 total_to_write = (U64)(opl - ptr);
|
|
DWORD to_write = total_to_write;
|
|
if (total_to_write > max_U32){
|
|
to_write = max_U32;
|
|
}
|
|
DWORD actual_write = 0;
|
|
if (!WriteFile(file, ptr, to_write, &actual_write, 0)){
|
|
result = 0;
|
|
goto dblbreak;
|
|
}
|
|
ptr += actual_write;
|
|
}
|
|
}
|
|
dblbreak:;
|
|
|
|
// close file
|
|
CloseHandle(file);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_write_list(String8 file_name, String8Node *first_node){
|
|
ProfBeginFunc();
|
|
|
|
// get handle
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 file_name16 = str16_from_str8(scratch.arena, file_name);
|
|
HANDLE file = CreateFileW((WCHAR*)file_name16.str,
|
|
GENERIC_WRITE, 0, 0,
|
|
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL,
|
|
0);
|
|
|
|
B32 result = 0;
|
|
if (file != INVALID_HANDLE_VALUE){
|
|
result = 1;
|
|
|
|
for (String8Node *node = first_node;
|
|
node != 0;
|
|
node = node->next){
|
|
U8 *ptr = node->string.str;
|
|
U8 *opl = ptr + node->string.size;
|
|
for (;ptr < opl;){
|
|
U64 total_to_write = (U64)(opl - ptr);
|
|
DWORD to_write = total_to_write;
|
|
if (total_to_write > max_U32){
|
|
to_write = max_U32;
|
|
}
|
|
DWORD actual_write = 0;
|
|
if (!WriteFile(file, ptr, to_write, &actual_write, 0)){
|
|
result = 0;
|
|
goto dblbreak;
|
|
}
|
|
ptr += actual_write;
|
|
}
|
|
}
|
|
dblbreak:;
|
|
|
|
CloseHandle(file);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL FileProperties
|
|
os_file_properties(String8 file_name){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 file_name16 = str16_from_str8(scratch.arena, file_name);
|
|
|
|
// get attribs and convert to properties
|
|
FileProperties result = {0};
|
|
WIN32_FILE_ATTRIBUTE_DATA attribs = {0};
|
|
if (GetFileAttributesExW((WCHAR*)file_name16.str, GetFileExInfoStandard,
|
|
&attribs)){
|
|
result.size = ((U64)attribs.nFileSizeHigh << 32) | (U64)attribs.nFileSizeLow;
|
|
result.flags = w32_prop_flags_from_attribs(attribs.dwFileAttributes);
|
|
result.create_time = w32_dense_time_from_file_time(&attribs.ftCreationTime);
|
|
result.modify_time = w32_dense_time_from_file_time(&attribs.ftLastWriteTime);
|
|
result.access = w32_access_from_attributes(attribs.dwFileAttributes);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_delete(String8 file_name){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 file_name16 = str16_from_str8(scratch.arena, file_name);
|
|
// delete file
|
|
B32 result = DeleteFileW((WCHAR*)file_name16.str);
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_rename(String8 og_name, String8 new_name){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 og_name16 = str16_from_str8(scratch.arena, og_name);
|
|
String16 new_name16 = str16_from_str8(scratch.arena, new_name);
|
|
// rename file
|
|
B32 result = MoveFileW((WCHAR*)og_name16.str, (WCHAR*)new_name16.str);
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_make_directory(String8 path){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 path16 = str16_from_str8(scratch.arena, path);
|
|
// make directory
|
|
B32 result = CreateDirectoryW((WCHAR*)path16.str, 0);
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_delete_directory(String8 path){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 path16 = str16_from_str8(scratch.arena, path);
|
|
// make directory
|
|
B32 result = RemoveDirectoryW((WCHAR*)path16.str);
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL OS_FileIter
|
|
os_file_iter_init(Arena *arena, String8 path){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
String8Node nodes[2];
|
|
String8List list;
|
|
|
|
nodes[0].string = path;
|
|
nodes[0].next = nodes + 1;
|
|
nodes[1].string = str8_lit("\\*");
|
|
nodes[1].next = 0;
|
|
|
|
list.first = nodes + 0;
|
|
list.last = nodes + 1;
|
|
list.node_count = 2;
|
|
list.total_size = path.size + 2;
|
|
|
|
ArenaTemp scratch = arena_get_scratch(&arena, 1);
|
|
String8 path_star = str8_join(scratch.arena, &list, 0);
|
|
// TODO(allen): Better unicode conversions here
|
|
String16 path16 = str16_from_str8(scratch.arena, path_star);
|
|
|
|
// store into iter
|
|
OS_FileIter result = {0};
|
|
W32_FileIter *w32_iter = (W32_FileIter*)&result;
|
|
w32_iter->handle = FindFirstFileW((WCHAR*)path16.str, &w32_iter->find_data);
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_iter_next(Arena *arena, OS_FileIter *iter,
|
|
String8 *name, FileProperties *props){
|
|
ProfBeginFunc();
|
|
|
|
B32 result = 0;
|
|
|
|
W32_FileIter *w32_iter = (W32_FileIter*)iter;
|
|
if (w32_iter->handle != 0 &&
|
|
w32_iter->handle != INVALID_HANDLE_VALUE){
|
|
for (;!w32_iter->done;){
|
|
// check for . and ..
|
|
WCHAR *file_name = w32_iter->find_data.cFileName;
|
|
B32 is_dot = (file_name[0] == '.' && file_name[1] == 0);
|
|
B32 is_dotdot = (file_name[0] == '.' && file_name[1] == '.' &&
|
|
file_name[2] == 0);
|
|
|
|
// setup to emit
|
|
B32 emit = (!is_dot && !is_dotdot);
|
|
WIN32_FIND_DATAW data = {0};
|
|
if (emit){
|
|
MemoryCopyStruct(&data, &w32_iter->find_data);
|
|
}
|
|
|
|
// increment the iterator
|
|
if (!FindNextFileW(w32_iter->handle, &w32_iter->find_data)){
|
|
w32_iter->done = 1;
|
|
}
|
|
|
|
// do the emit if we saved one earlier
|
|
if (emit){
|
|
*name = str8_from_str16(arena, str16_cstring((U16*)data.cFileName));
|
|
props->size = ((U64)data.nFileSizeHigh << 32) | (U64)data.nFileSizeLow;
|
|
props->flags = w32_prop_flags_from_attribs(data.dwFileAttributes);
|
|
props->create_time = w32_dense_time_from_file_time(&data.ftCreationTime);
|
|
props->modify_time = w32_dense_time_from_file_time(&data.ftLastWriteTime);
|
|
props->access = w32_access_from_attributes(data.dwFileAttributes);
|
|
result = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_file_iter_end(OS_FileIter *iter){
|
|
ProfBeginFunc();
|
|
|
|
W32_FileIter *w32_iter = (W32_FileIter*)iter;
|
|
if (w32_iter->handle != 0 &&
|
|
w32_iter->handle != INVALID_HANDLE_VALUE){
|
|
FindClose(w32_iter->handle);
|
|
}
|
|
|
|
ProfEndFunc();
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
os_file_path(Arena *arena, OS_SystemPath path){
|
|
ProfBeginFunc();
|
|
|
|
String8 result = {0};
|
|
|
|
switch (path){
|
|
case OS_SystemPath_CurrentDirectory:
|
|
{
|
|
ArenaTemp scratch = arena_get_scratch(&arena, 1);
|
|
DWORD cap = 2048;
|
|
U16 *buffer = push_array(scratch.arena, U16, cap);
|
|
DWORD size = GetCurrentDirectoryW(cap, (WCHAR*)buffer);
|
|
if (size >= cap){
|
|
arena_end_temp(&scratch);
|
|
buffer = push_array(scratch.arena, U16, size + 1);
|
|
size = GetCurrentDirectoryW(size + 1, (WCHAR*)buffer);
|
|
}
|
|
result = str8_from_str16(arena, CLiteral(String16){buffer, size});
|
|
arena_release_scratch(&scratch);
|
|
}break;
|
|
|
|
case OS_SystemPath_Binary:
|
|
{
|
|
result = str8_push_copy(arena, w32_binary_path);
|
|
}break;
|
|
|
|
case OS_SystemPath_UserData:
|
|
{
|
|
result = str8_push_copy(arena, w32_user_path);
|
|
}break;
|
|
|
|
case OS_SystemPath_TempData:
|
|
{
|
|
result = str8_push_copy(arena, w32_temp_path);
|
|
}break;
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_set_current_directory(String8 path){
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
|
|
String16 path16 = str16_from_str8(scratch.arena, path);
|
|
SetCurrentDirectoryW((WCHAR*)path16.str);
|
|
|
|
arena_release_scratch(&scratch);
|
|
}
|
|
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: Time
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
os_now_universal_time(void){
|
|
SYSTEMTIME system_time = {0};
|
|
GetSystemTime(&system_time);
|
|
DateTime result = w32_date_time_from_system_time(&system_time);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
os_local_time_from_universal(DateTime *univ_date_time){
|
|
SYSTEMTIME univ_system_time = w32_system_time_from_date_time(univ_date_time);
|
|
FILETIME univ_file_time = {0};
|
|
SystemTimeToFileTime(&univ_system_time, &univ_file_time);
|
|
FILETIME local_file_time = {0};
|
|
FileTimeToLocalFileTime(&univ_file_time, &local_file_time);
|
|
SYSTEMTIME local_system_time = {0};
|
|
FileTimeToSystemTime(&local_file_time, &local_system_time);
|
|
DateTime result = w32_date_time_from_system_time(&local_system_time);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
os_universal_time_from_local(DateTime *local_date_time){
|
|
SYSTEMTIME local_system_time = w32_system_time_from_date_time(local_date_time);
|
|
FILETIME local_file_time = {0};
|
|
SystemTimeToFileTime(&local_system_time, &local_file_time);
|
|
FILETIME univ_file_time = {0};
|
|
LocalFileTimeToFileTime(&local_file_time, &univ_file_time);
|
|
SYSTEMTIME univ_system_time = {0};
|
|
FileTimeToSystemTime(&univ_file_time, &univ_system_time);
|
|
DateTime result = w32_date_time_from_system_time(&univ_system_time);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL U32
|
|
os_time_stamp_32_from_date_time(DateTime *date_time){
|
|
SYSTEMTIME system_time = w32_system_time_from_date_time(date_time);
|
|
FILETIME file_time = {0};
|
|
SystemTimeToFileTime(&system_time, &file_time);
|
|
U64 file_time64 = ( (U64) file_time.dwLowDateTime +
|
|
((U64)(file_time.dwHighDateTime) << 32ll)
|
|
);
|
|
U32 result = (U32) ((file_time64 - 116444736000000000ll)/10000000ll);
|
|
return(result);
|
|
}
|
|
|
|
|
|
MR4TH_SYMBOL U64
|
|
os_now_ticks(void){
|
|
U64 result = 0;
|
|
LARGE_INTEGER perf_counter = {0};
|
|
if (QueryPerformanceCounter(&perf_counter)){
|
|
result = ((U64)perf_counter.HighPart << 32) | perf_counter.LowPart;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_get_microseconds_per_tick(U64 *usecs_out, U64 *ticks_out){
|
|
*usecs_out = Million(1);
|
|
*ticks_out = w32_ticks_per_second;
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_sleep_milliseconds(U32 t){
|
|
Sleep(t);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: Libraries
|
|
|
|
MR4TH_SYMBOL OS_Library*
|
|
os_lib_load(String8 path){
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String16 path16 = str16_from_str8(scratch.arena, path);
|
|
OS_Library *result = (OS_Library*)(LoadLibraryW((WCHAR*)path16.str));
|
|
arena_release_scratch(&scratch);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL VoidFunc*
|
|
os_lib_get_proc(OS_Library *lib, char *name){
|
|
HMODULE module = (HMODULE)(lib);
|
|
VoidFunc *result = (VoidFunc*)(GetProcAddress(module, name));
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_lib_release(OS_Library *lib){
|
|
HMODULE module = (HMODULE)(lib);
|
|
FreeLibrary(module);
|
|
}
|
|
|
|
MR4TH_SYMBOL OS_Library*
|
|
os_lib_from_addr(void *addr){
|
|
HMODULE module = 0;
|
|
GetModuleHandleExW(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS,
|
|
(LPWSTR)addr, &module);
|
|
return((OS_Library*)module);
|
|
}
|
|
|
|
MR4TH_SYMBOL OS_VRange
|
|
os_lib_image_range(OS_Library *lib){
|
|
OS_VRange result = {0};
|
|
HANDLE process = GetCurrentProcess();
|
|
MODULEINFO module_info = {0};
|
|
if (GetModuleInformation(process, (HMODULE)lib,
|
|
&module_info, sizeof(module_info))){
|
|
result.addr = (U8*)module_info.lpBaseOfDll;
|
|
result.size = module_info.SizeOfImage;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Implementation: Entropy
|
|
|
|
MR4TH_SYMBOL void
|
|
os_get_entropy(void *data, U64 size){
|
|
HCRYPTPROV prov = 0;
|
|
CryptAcquireContextW(&prov, 0, 0, PROV_DSS, CRYPT_VERIFYCONTEXT);
|
|
CryptGenRandom(prov, size, (BYTE*)data);
|
|
CryptReleaseContext(prov, 0);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Functions: Specialized Init for WinMain
|
|
|
|
MR4TH_SYMBOL_STATIC HINSTANCE w32_instance = 0;
|
|
|
|
MR4TH_SYMBOL void
|
|
w32_WinMain_init(HINSTANCE hInstance,
|
|
HINSTANCE hPrevInstance,
|
|
LPSTR lpCmdLine,
|
|
int nShowCmd){
|
|
int argc = __argc;
|
|
char **argv = __argv;
|
|
w32_instance = hInstance;
|
|
os_main_init(argc, argv);
|
|
}
|
|
|
|
MR4TH_SYMBOL HINSTANCE
|
|
w32_get_instance(void){
|
|
if (w32_instance == 0){
|
|
w32_instance = GetModuleHandle(0);
|
|
}
|
|
return(w32_instance);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Functions: Time Helpers
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
w32_date_time_from_system_time(SYSTEMTIME *in){
|
|
DateTime result = {0};
|
|
result.year = in->wYear;
|
|
result.mon = (U8)in->wMonth;
|
|
result.day = in->wDay;
|
|
result.hour = in->wHour;
|
|
result.min = in->wMinute;
|
|
result.sec = in->wSecond;
|
|
result.msec = in->wMilliseconds;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL SYSTEMTIME
|
|
w32_system_time_from_date_time(DateTime *in){
|
|
SYSTEMTIME result = {0};
|
|
result.wYear = in->year;
|
|
result.wMonth = in->mon;
|
|
result.wDay = in->day;
|
|
result.wHour = in->hour;
|
|
result.wMinute = in->min;
|
|
result.wSecond = in->sec;
|
|
result.wMilliseconds = in->msec;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DenseTime
|
|
w32_dense_time_from_file_time(FILETIME *file_time){
|
|
SYSTEMTIME system_time = {0};
|
|
FileTimeToSystemTime(file_time, &system_time);
|
|
DateTime date_time = w32_date_time_from_system_time(&system_time);
|
|
DenseTime result = dense_time_from_date_time(&date_time);
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Win32 Functions: File Helpers
|
|
|
|
MR4TH_SYMBOL FilePropertyFlags
|
|
w32_prop_flags_from_attribs(DWORD attribs){
|
|
FilePropertyFlags result = 0;
|
|
if (attribs & FILE_ATTRIBUTE_DIRECTORY){
|
|
result |= FilePropertyFlag_IsFolder;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DataAccessFlags
|
|
w32_access_from_attributes(DWORD attribs){
|
|
DataAccessFlags result = DataAccessFlag_Read|DataAccessFlag_Execute;
|
|
if (!(attribs & FILE_ATTRIBUTE_READONLY)){
|
|
result |= DataAccessFlag_Write;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
#endif /* OS_WINDOWS */
|
|
|
|
/*******************************
|
|
** End Base Win32 **
|
|
*******************************/
|
|
|
|
|
|
|
|
/*******************************
|
|
** Begin Base Linux **
|
|
*******************************/
|
|
|
|
#if OS_LINUX
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: Global Variables
|
|
|
|
MR4TH_SYMBOL_STATIC Arena *lnx_perm_arena = 0;
|
|
MR4TH_SYMBOL_STATIC String8 lnx_binary_path = {0};
|
|
MR4TH_SYMBOL_STATIC String8 lnx_user_path = {0};
|
|
MR4TH_SYMBOL_STATIC String8 lnx_temp_path = {0};
|
|
|
|
MR4TH_SYMBOL_STATIC String8List lnx_cmd_line = {0};
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: Process Setup
|
|
|
|
MR4TH_SYMBOL void
|
|
os_main_init(int argc, char **argv){
|
|
ProfBeginFunc();
|
|
|
|
// arena
|
|
lnx_perm_arena = arena_alloc();
|
|
|
|
// command line arguments
|
|
for (int i = 0; i < argc; i += 1){
|
|
String8 arg = str8_cstring((U8*)argv[i]);
|
|
str8_list_push(lnx_perm_arena, &lnx_cmd_line, arg);
|
|
}
|
|
|
|
// paths
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
|
|
// binary path
|
|
{
|
|
B32 got_final_result = 0;
|
|
U8 *buffer = 0;
|
|
int size = 0;
|
|
for (U64 cap = PATH_MAX, r = 0; r < 4; cap *= 2, r += 1){
|
|
arena_end_temp(&scratch);
|
|
buffer = push_array_no_zero(scratch.arena, U8, cap);
|
|
size = readlink("/proc/self/exe", (char*)buffer, cap);
|
|
if (size < cap){
|
|
got_final_result = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (got_final_result && size > 0){
|
|
String8 full_name = str8(buffer, (U64)size);
|
|
String8 name_chopped = str8_chop_last_slash(full_name);
|
|
lnx_binary_path = str8_push_copy(lnx_perm_arena, name_chopped);
|
|
}
|
|
}
|
|
|
|
// user data path
|
|
{
|
|
char *home = getenv("HOME");
|
|
lnx_user_path = str8_cstring((U8*)home);
|
|
}
|
|
|
|
// temp data path
|
|
{
|
|
lnx_temp_path = str8_lit("/var/tmp");
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
}
|
|
|
|
MR4TH_SYMBOL String8List
|
|
os_command_line_arguments(void){
|
|
String8List result = lnx_cmd_line;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_exit_process(U32 code){
|
|
ProfClose();
|
|
exit((int)code);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: Memory Functions
|
|
|
|
MR4TH_SYMBOL void*
|
|
os_memory_reserve(U64 size){
|
|
void *result = mmap(0, size, PROT_NONE, MAP_PRIVATE|MAP_ANONYMOUS, -1, 0);
|
|
if (result == MAP_FAILED){
|
|
result = 0;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_memory_commit(void *ptr, U64 size){
|
|
B32 result = 0;
|
|
if (mprotect(ptr, size, PROT_READ|PROT_WRITE) == 0){
|
|
result = 1;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_memory_decommit(void *ptr, U64 size){
|
|
madvise(ptr, size, MADV_DONTNEED);
|
|
mprotect(ptr, size, PROT_NONE);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_memory_release(void *ptr, U64 size){
|
|
munmap(ptr, size);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_memory_protect(void *ptr, U64 size, DataAccessFlags access_flags){
|
|
int prot = PROT_NONE;
|
|
if ((access_flags&DataAccessFlag_Read) == 0){
|
|
prot |= PROT_READ;
|
|
}
|
|
if ((access_flags&DataAccessFlag_Write) == 0){
|
|
prot |= PROT_WRITE;
|
|
}
|
|
if ((access_flags&DataAccessFlag_Execute) == 0){
|
|
prot |= PROT_EXEC;
|
|
}
|
|
|
|
B32 result = (mprotect(ptr, size, prot));
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: File Handling
|
|
|
|
MR4TH_SYMBOL String8
|
|
os_file_read(Arena *arena, String8 file_name){
|
|
ProfBeginFunc();
|
|
|
|
// get handle
|
|
ArenaTemp scratch = arena_get_scratch(&arena, 1);
|
|
// (this ensures file_name8 has a null terminator)
|
|
String8 file_name8 = str8_push_copy(scratch.arena, file_name);
|
|
int file = open((char*)file_name8.str, O_RDONLY);
|
|
|
|
String8 result = {0};
|
|
if (file >= 0){
|
|
// get size
|
|
struct stat file_stat = {0};
|
|
U64 size = 0;
|
|
if (fstat(file, &file_stat) == 0){
|
|
size = file_stat.st_size;
|
|
}
|
|
|
|
// allocate buffer
|
|
if (size > 0){
|
|
ArenaTemp restore_point = arena_begin_temp(arena);
|
|
U8 *buffer = push_array_no_zero(arena, U8, size + 1);
|
|
|
|
// read
|
|
U8 *ptr = buffer;
|
|
U8 *opl = buffer + size;
|
|
B32 success = 1;
|
|
for (;ptr < opl;){
|
|
U64 total_to_read = (U64)(opl - ptr);
|
|
U32 to_read = (U32)total_to_read;
|
|
if (total_to_read > max_U32){
|
|
to_read = max_U32;
|
|
}
|
|
int read_result = read(file, buffer, to_read);
|
|
if (read_result < 0){
|
|
success = 0;
|
|
break;
|
|
}
|
|
else if (read_result == 0){
|
|
break;
|
|
}
|
|
ptr += read_result;
|
|
}
|
|
|
|
// set result or reset memory
|
|
if (success){
|
|
U64 final_size = (U64)(ptr - buffer);
|
|
final_size = ClampTop(final_size, size);
|
|
buffer[final_size] = 0;
|
|
result.str = buffer;
|
|
result.size = final_size;
|
|
}
|
|
else{
|
|
arena_end_temp(&restore_point);
|
|
}
|
|
}
|
|
|
|
// close file
|
|
close(file);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_write_callback(String8 file_name, void *udata, OS_FileWriteCallback *callback){
|
|
ProfBeginFunc();
|
|
|
|
// get handle
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 file_name8 = str8_push_copy(scratch.arena, file_name);
|
|
int file = open((char*)file_name8.str, O_CREAT|O_WRONLY|O_TRUNC, 0666);
|
|
|
|
B32 result = 0;
|
|
if (file >= 0){
|
|
result = 1;
|
|
|
|
for (;;){
|
|
String8 chunk = callback(udata);
|
|
if (chunk.size == 0){
|
|
break;
|
|
}
|
|
|
|
U8 *ptr = chunk.str;
|
|
U8 *opl = chunk.str + chunk.size;
|
|
for (;ptr < opl;){
|
|
U64 total_to_write = (U64)(opl - ptr);
|
|
U32 to_write = total_to_write;
|
|
if (total_to_write > (U32)SSIZE_MAX){
|
|
to_write = (U32)SSIZE_MAX;
|
|
}
|
|
int write_result = write(file, ptr, to_write);
|
|
|
|
if (write_result <= 0){
|
|
result = 0;
|
|
goto dblbreak;
|
|
}
|
|
ptr += write_result;
|
|
}
|
|
|
|
}
|
|
dblbreak:;
|
|
|
|
// close file
|
|
close(file);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL FileProperties
|
|
os_file_properties(String8 file_name){
|
|
ProfBeginFunc();
|
|
|
|
// convert name
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
// (this ensures file_name8 has a null terminator)
|
|
String8 file_name8 = str8_push_copy(scratch.arena, file_name);
|
|
int file = open((char*)file_name8.str, O_RDONLY);
|
|
|
|
// get file stat and convert to properties
|
|
FileProperties result = {0};
|
|
struct stat file_stat = {0};
|
|
if (fstat(file, &file_stat) == 0){
|
|
result = lnx_file_properties_from_stat(&file_stat);
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_delete(String8 file_name){
|
|
ProfBeginFunc();
|
|
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 file_name8 = str8_push_copy(scratch.arena, file_name);
|
|
|
|
B32 result = 0;
|
|
if (remove((char*)file_name8.str) >= 0){
|
|
result = 1;
|
|
}
|
|
|
|
arena_end_temp(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_rename(String8 og_name, String8 new_name){
|
|
ProfBeginFunc();
|
|
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 og_name8 = str8_push_copy(scratch.arena, og_name);
|
|
String8 new_name8 = str8_push_copy(scratch.arena, new_name);
|
|
|
|
int rename_result = rename((char*)og_name8.str, (char*)new_name8.str);
|
|
B32 result = (rename_result >= 0);
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_make_directory(String8 path){
|
|
ProfBeginFunc();
|
|
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 path8 = str8_push_copy(scratch.arena, path);
|
|
|
|
B32 result = 0;
|
|
if (mkdir((char*)path8.str, 0755) >= 0){
|
|
result = 1;
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_delete_directory(String8 path){
|
|
ProfBeginFunc();
|
|
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 path8 = str8_push_copy(scratch.arena, path);
|
|
|
|
B32 result = 0;
|
|
if (rmdir((char*)path8.str) >= 0){
|
|
result = 1;
|
|
}
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL OS_FileIter
|
|
os_file_iter_init(Arena *arena, String8 path){
|
|
ProfBeginFunc();
|
|
|
|
String8 path8 = str8_push_copy(arena, path);
|
|
|
|
OS_FileIter result = {0};
|
|
LNX_FileIter *lnx_iter = (LNX_FileIter*)&result;
|
|
lnx_iter->dir = opendir((char*)path8.str);
|
|
lnx_iter->path = path8;
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL B32
|
|
os_file_iter_next(Arena *arena, OS_FileIter *iter,
|
|
String8 *name, FileProperties *props){
|
|
B32 result = 0;
|
|
LNX_FileIter *lnx_iter = (LNX_FileIter*)iter;
|
|
|
|
String8 path = lnx_iter->path;
|
|
|
|
for (;lnx_iter->dir != 0;){
|
|
struct dirent *dp = readdir(lnx_iter->dir);
|
|
result = (dp != 0);
|
|
|
|
String8 file_name = {0};
|
|
if (result){
|
|
file_name = str8_cstring((U8*)dp->d_name);
|
|
}
|
|
|
|
struct stat st = {0};
|
|
int stat_result = 0;
|
|
if (result){
|
|
ArenaTemp scratch = arena_get_scratch(&arena, 1);
|
|
|
|
String8 full_path = {0};
|
|
full_path.size = path.size + 1 + file_name.size;
|
|
full_path.str = push_array_no_zero(scratch.arena, U8, full_path.size + 1);
|
|
MemoryCopy(full_path.str, path.str, path.size);
|
|
full_path.str[path.size] = '/';
|
|
MemoryCopy(full_path.str, file_name.str, file_name.size);
|
|
full_path.str[full_path.size] = 0;
|
|
|
|
stat_result = stat((char*)full_path.str, &st);
|
|
|
|
arena_release_scratch(&scratch);
|
|
}
|
|
|
|
B32 filtered = 0;
|
|
if (result){
|
|
if (file_name.str != 0 &&
|
|
((file_name.str[0] == '.' && file_name.str[1] == 0) ||
|
|
(file_name.str[0] == '.' && file_name.str[1] == 0 && file_name.str[2] == 0))){
|
|
filtered = 1;
|
|
}
|
|
}
|
|
|
|
if (result && !filtered){
|
|
*name = str8_push_copy(arena, file_name);
|
|
if (stat_result >= 0){
|
|
*props = lnx_file_properties_from_stat(&st);
|
|
}
|
|
result = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_file_iter_end(OS_FileIter *iter){
|
|
ProfBeginFunc();
|
|
|
|
LNX_FileIter *lnx_iter = (LNX_FileIter*)iter;
|
|
if (lnx_iter->dir != 0){
|
|
closedir(lnx_iter->dir);
|
|
lnx_iter->dir = 0;
|
|
}
|
|
|
|
ProfEndFunc();
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
os_file_path(Arena *arena, OS_SystemPath path){
|
|
ProfBeginFunc();
|
|
|
|
String8 result = {0};
|
|
|
|
switch (path){
|
|
default:break;
|
|
|
|
case OS_SystemPath_CurrentDirectory:
|
|
{
|
|
char *cwd = getcwd(0, 0);
|
|
result = str8_push_copy(arena, str8_cstring((U8*)cwd));
|
|
free(cwd);
|
|
}break;
|
|
|
|
case OS_SystemPath_Binary:
|
|
{
|
|
result = str8_push_copy(arena, lnx_binary_path);
|
|
}break;
|
|
|
|
case OS_SystemPath_UserData:
|
|
{
|
|
result = str8_push_copy(arena, lnx_user_path);
|
|
}break;
|
|
|
|
case OS_SystemPath_TempData:
|
|
{
|
|
result = str8_push_copy(arena, lnx_temp_path);
|
|
}break;
|
|
}
|
|
|
|
ProfEndFunc();
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_set_current_directory(String8 path){
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 path8 = str8_push_copy(scratch.arena, path);
|
|
chdir((char*)path8.str);
|
|
arena_release_scratch(&scratch);
|
|
}
|
|
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: Time
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
os_now_universal_time(void){
|
|
time_t t = time(0);
|
|
struct tm tm = {0};
|
|
struct tm *tmptr = gmtime_r(&t, &tm);
|
|
DateTime result = {0};
|
|
if (tmptr != 0){
|
|
result = lnx_date_time_from_tm(&tm);
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
os_local_time_from_universal(DateTime *date_time){
|
|
struct tm univ_tm = lnx_tm_from_date_time(date_time);
|
|
univ_tm.tm_isdst = -1;
|
|
time_t univ_time_t = timegm(&univ_tm);
|
|
struct tm local_tm = {0};
|
|
localtime_r(&univ_time_t, &local_tm);
|
|
DateTime result = lnx_date_time_from_tm(&local_tm);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
os_universal_time_from_local(DateTime *date_time){
|
|
struct tm local_tm = lnx_tm_from_date_time(date_time);
|
|
local_tm.tm_isdst = -1;
|
|
time_t univ_time_t = mktime(&local_tm);
|
|
struct tm univ_tm = {0};
|
|
gmtime_r(&univ_time_t, &univ_tm);
|
|
DateTime result = lnx_date_time_from_tm(&univ_tm);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL U32
|
|
os_time_stamp_32_from_date_time(DateTime *date_time){
|
|
struct tm tm = lnx_tm_from_date_time(date_time);
|
|
time_t t = timegm(&tm);
|
|
return((U32)t);
|
|
}
|
|
|
|
MR4TH_SYMBOL U64
|
|
os_now_ticks(void){
|
|
struct timespec t = {0};
|
|
clock_gettime(CLOCK_MONOTONIC, &t);
|
|
U64 result = t.tv_sec*Million(1) + t.tv_nsec/Thousand(1);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_get_microseconds_per_tick(U64 *usecs_out, U64 *ticks_out){
|
|
*usecs_out = 1;
|
|
*ticks_out = 1;
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_sleep_milliseconds(U32 t){
|
|
usleep(t*Thousand(1));
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: Libraries
|
|
|
|
MR4TH_SYMBOL OS_Library*
|
|
os_lib_load(String8 path){
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
String8 path8 = str8_push_copy(scratch.arena, path);
|
|
OS_Library *result = (OS_Library*)dlopen((char*)path8.str, RTLD_LAZY|RTLD_LOCAL);
|
|
arena_release_scratch(&scratch);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL VoidFunc*
|
|
os_lib_get_proc(OS_Library *lib, char *name){
|
|
VoidFunc *proc = (VoidFunc*)dlsym((void*)lib, name);
|
|
return(proc);
|
|
}
|
|
|
|
MR4TH_SYMBOL void
|
|
os_lib_release(OS_Library *lib){
|
|
dlclose((void*)lib);
|
|
}
|
|
|
|
MR4TH_SYMBOL OS_Library*
|
|
os_lib_from_addr(void *addr){
|
|
UAddr addrx = IntFromPtr(addr);
|
|
ArenaTemp scratch = arena_get_scratch(0, 0);
|
|
|
|
// NOTE(allen):
|
|
// Implementing this on Linux is a lot less solidly
|
|
// supported than it is on Windows.
|
|
//
|
|
// As far as I can tell, reading "/proc/self/maps" is the
|
|
// only portable way to get at the information. The fact
|
|
// that this information is in only available in this text
|
|
// based unaccelerated structured is unfortunate.
|
|
//
|
|
// But even worse is the fact that, even with this info,
|
|
// to actually get one of the opaque handles to the
|
|
// library, we have to call dlopen again -- no other API
|
|
// appears to exist which can return the required information.
|
|
//
|
|
// But that also means that the library gets a reference
|
|
// count from this call and so we have to then call dlclose
|
|
// on the handle just to decrement that reference counter.
|
|
//
|
|
// I don't even know yet if this handles the base executable
|
|
// of the program correctly.
|
|
|
|
// read "/proc/self/maps"
|
|
B32 got_result = 0;
|
|
U8 *buffer = 0;
|
|
int size = 0;
|
|
for (U64 cap = KB(4), r = 0; r < 6; cap *= 64, r += 1){
|
|
arena_end_temp(&scratch);
|
|
buffer = push_array_no_zero(scratch.arena, U8, cap);
|
|
size = readlink("/proc/self/maps", (char*)buffer, cap);
|
|
if (size < cap){
|
|
got_result = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// parse contents from "/proc/self/maps"
|
|
String8 contents = str8(buffer, (U64)size);
|
|
String8List lines = str8_split(scratch.arena, contents, (U8*)"\n", 1);
|
|
for (String8Node *node = lines.first;
|
|
node != 0;
|
|
node = node->next){
|
|
String8 line = node->string;
|
|
U8 *opl = line.str + line.size;
|
|
|
|
|
|
U8 *dash_ptr = line.str;
|
|
for (;dash_ptr < opl && *dash_ptr != '-'; dash_ptr += 1);
|
|
|
|
U8 *addr_max_opl = dash_ptr;
|
|
for (;dash_ptr < opl && *dash_ptr != ' '; dash_ptr += 1);
|
|
|
|
U8 *slash_ptr = addr_max_opl;
|
|
for (;slash_ptr < opl && *slash_ptr != '/'; slash_ptr += 1);
|
|
|
|
U64 addr_min = u64_from_str8(str8_range(line.str, dash_ptr), 0x10);
|
|
U64 addr_max = u64_from_str8(str8_range(dash_ptr + 1, addr_max_opl), 0x10);
|
|
String8 path = str8_range(slash_ptr, opl);
|
|
|
|
|
|
}
|
|
|
|
push_array_no_zero(scratch.arena, U8, size);
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
OS_Library *library = 0;
|
|
return(library);
|
|
}
|
|
|
|
MR4TH_SYMBOL OS_VRange
|
|
os_lib_image_range(OS_Library *lib){
|
|
OS_VRange result = {0};
|
|
// TODO(allen):
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Linux Implementation: Entropy
|
|
|
|
MR4TH_SYMBOL void
|
|
os_get_entropy(void *data, U64 size){
|
|
getrandom(data, size, 0);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Linux Functions: Time Helpers
|
|
|
|
MR4TH_SYMBOL DateTime
|
|
lnx_date_time_from_tm(struct tm *tm){
|
|
DateTime result = {0};
|
|
result.year = tm->tm_year + 1900;
|
|
result.mon = tm->tm_mon + 1;
|
|
result.day = tm->tm_mday;
|
|
result.hour = tm->tm_hour;
|
|
result.min = tm->tm_min;
|
|
result.sec = tm->tm_sec;
|
|
result.msec = 0;
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DenseTime
|
|
lnx_dense_time_from_time_t(time_t time){
|
|
struct tm system_time = *gmtime(&time);
|
|
DateTime date_time = lnx_date_time_from_tm(&system_time);
|
|
DenseTime result = dense_time_from_date_time(&date_time);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL struct tm
|
|
lnx_tm_from_date_time(DateTime *date_time){
|
|
struct tm result = {0};
|
|
result.tm_sec = date_time->sec;
|
|
result.tm_min = date_time->min;
|
|
result.tm_hour = date_time->hour;
|
|
result.tm_mday = date_time->day;
|
|
result.tm_mon = date_time->mon - 1;
|
|
result.tm_year = date_time->year;
|
|
return(result);
|
|
}
|
|
|
|
////////////////////////////////
|
|
// Linux Functions: File Helpers
|
|
|
|
MR4TH_SYMBOL FilePropertyFlags
|
|
lnx_prop_flags_from_mode(mode_t mode){
|
|
FilePropertyFlags result = 0;
|
|
if ((mode&S_IFDIR) != 0){
|
|
result |= FilePropertyFlag_IsFolder;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL DataAccessFlags
|
|
lnx_access_from_mode(mode_t mode){
|
|
DataAccessFlags result = 0;
|
|
if ((mode&(S_IRUSR|S_IRGRP)) != 0){
|
|
result |= DataAccessFlag_Read;
|
|
}
|
|
if ((mode&(S_IWUSR|S_IWGRP)) != 0){
|
|
result |= DataAccessFlag_Write;
|
|
}
|
|
if ((mode&(S_IXUSR|S_IXGRP)) != 0){
|
|
result |= DataAccessFlag_Execute;
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL FileProperties
|
|
lnx_file_properties_from_stat(struct stat *st){
|
|
FileProperties result = {0};
|
|
result.size = st->st_size;
|
|
result.flags = lnx_prop_flags_from_mode(st->st_mode);
|
|
result.create_time = lnx_dense_time_from_time_t(st->st_ctime);
|
|
result.modify_time = lnx_dense_time_from_time_t(st->st_mtime);
|
|
// TODO(allen): research:
|
|
// how do these flags from fstat behave?
|
|
// S_IRUSR, S_IWUSR, S_IXUSR
|
|
// S_IRGRP, S_IWGRP, S_IXGRP
|
|
// S_IROTH, S_IWOTH, S_IXOTH
|
|
result.access = lnx_access_from_mode(st->st_mode);
|
|
return(result);
|
|
}
|
|
|
|
MR4TH_SYMBOL String8
|
|
lnx_proc_file_read(Arena *arena, char *path){
|
|
ArenaTemp scratch = arena_get_scratch(&arena, 1);
|
|
|
|
int file = open(path, O_RDONLY);
|
|
|
|
String8List list = {0};
|
|
if (file >= 0){
|
|
S64 cap = KB(4);
|
|
for (;;){
|
|
U8 *buffer = push_array_no_zero(scratch.arena, U8, cap);
|
|
int size = read(file, buffer, cap);
|
|
if (size > 0){
|
|
str8_list_push(scratch.arena, &list, str8(buffer, (U64)size));
|
|
}
|
|
if (size < cap){
|
|
break;
|
|
}
|
|
cap *= 4;
|
|
}
|
|
}
|
|
|
|
String8 result = str8_join(arena, &list, 0);
|
|
|
|
arena_release_scratch(&scratch);
|
|
|
|
return(result);
|
|
}
|
|
|
|
#endif /* OS_LINUX */
|
|
|
|
/*******************************
|
|
** End Base Linux **
|
|
*******************************/
|
|
|
|
|
|
////////////////////////////////////////////////
|
|
////////////////////////////////////////////////
|
|
///////////////// M4 Printf ////////////////
|
|
////////////////////////////////////////////////
|
|
////////////////////////////////////////////////
|
|
|
|
// stb_sprintf.h STB_SPRINTF_IMPLEMENTATION
|
|
|
|
#define stbsp__uint32 U32
|
|
#define stbsp__int32 S32
|
|
#define stbsp__uint64 U64
|
|
#define stbsp__int64 S64
|
|
#define stbsp__uint16 U16
|
|
#define stbsp__uintptr UAddr
|
|
|
|
// NOTE: modification - removed STB_SPRINTF_MSVC_MODE -- compatibility mode for behavior from MSVC2013 and earlier
|
|
|
|
#ifdef STB_SPRINTF_NOUNALIGNED // define this before inclusion to force stbsp_sprintf to always use aligned accesses
|
|
#define STBSP__UNALIGNED(code)
|
|
#else
|
|
#define STBSP__UNALIGNED(code) code
|
|
#endif
|
|
|
|
// internal float utility functions
|
|
static stbsp__int32 stbsp__real_to_str(char const **start, stbsp__uint32 *len, char *out, stbsp__int32 *decimal_pos, double value, stbsp__uint32 frac_digits);
|
|
static stbsp__int32 stbsp__real_to_parts(stbsp__int64 *bits, stbsp__int32 *expo, double value);
|
|
#define STBSP__SPECIAL 0x7000
|
|
|
|
static char stbsp__period = '.';
|
|
static char stbsp__comma = ',';
|
|
static struct{
|
|
short temp; // force next field to be 2-byte aligned
|
|
char pair[201];
|
|
} stbsp__digitpair =
|
|
{
|
|
0,
|
|
"00010203040506070809101112131415161718192021222324"
|
|
"25262728293031323334353637383940414243444546474849"
|
|
"50515253545556575859606162636465666768697071727374"
|
|
"75767778798081828384858687888990919293949596979899"
|
|
};
|
|
|
|
STBSP__PUBLICDEF void m4_set_separators(char pcomma, char pperiod)
|
|
{
|
|
stbsp__period = pperiod;
|
|
stbsp__comma = pcomma;
|
|
}
|
|
|
|
#define STBSP__LEFTJUST 1
|
|
#define STBSP__LEADINGPLUS 2
|
|
#define STBSP__LEADINGSPACE 4
|
|
#define STBSP__LEADING_0X 8
|
|
#define STBSP__LEADINGZERO 16
|
|
#define STBSP__INTMAX 32
|
|
#define STBSP__TRIPLET_COMMA 64
|
|
#define STBSP__NEGATIVE 128
|
|
#define STBSP__MEMORY_SIZES 256
|
|
#define STBSP__HALFWIDTH 512
|
|
|
|
static void stbsp__lead_sign(stbsp__uint32 fl, char *sign)
|
|
{
|
|
sign[0] = 0;
|
|
if (fl & STBSP__NEGATIVE) {
|
|
sign[0] = 1;
|
|
sign[1] = '-';
|
|
} else if (fl & STBSP__LEADINGSPACE) {
|
|
sign[0] = 1;
|
|
sign[1] = ' ';
|
|
} else if (fl & STBSP__LEADINGPLUS) {
|
|
sign[0] = 1;
|
|
sign[1] = '+';
|
|
}
|
|
}
|
|
|
|
static STBSP__ASAN stbsp__uint32 stbsp__strlen_limited(char const *s, stbsp__uint32 limit)
|
|
{
|
|
char const * sn = s;
|
|
|
|
// get up to 4-byte alignment
|
|
for (;;) {
|
|
if (((stbsp__uintptr)sn & 3) == 0)
|
|
break;
|
|
|
|
if (!limit || *sn == 0)
|
|
return (stbsp__uint32)(sn - s);
|
|
|
|
++sn;
|
|
--limit;
|
|
}
|
|
|
|
// scan over 4 bytes at a time to find terminating 0
|
|
// this will intentionally scan up to 3 bytes past the end of buffers,
|
|
// but becase it works 4B aligned, it will never cross page boundaries
|
|
// (hence the STBSP__ASAN markup; the over-read here is intentional
|
|
// and harmless)
|
|
while (limit >= 4) {
|
|
stbsp__uint32 v = *(stbsp__uint32 *)sn;
|
|
// bit hack to find if there's a 0 byte in there
|
|
if ((v - 0x01010101) & (~v) & 0x80808080UL)
|
|
break;
|
|
|
|
sn += 4;
|
|
limit -= 4;
|
|
}
|
|
|
|
// handle the last few characters to find actual size
|
|
while (limit && *sn) {
|
|
++sn;
|
|
--limit;
|
|
}
|
|
|
|
return (stbsp__uint32)(sn - s);
|
|
}
|
|
|
|
STBSP__PUBLICDEF int m4_vsprintfcb(STBSP_SPRINTFCB *callback, void *user, char *buf, char const *fmt, va_list va)
|
|
{
|
|
static char hex[] = "0123456789abcdefxp";
|
|
static char hexu[] = "0123456789ABCDEFXP";
|
|
static char spaces[] =
|
|
" "
|
|
" "
|
|
" "
|
|
" ";
|
|
char *bf = buf;
|
|
char const *f = fmt;
|
|
int tlen = 0;
|
|
|
|
for (;;){
|
|
stbsp__int32 field_width;
|
|
stbsp__int32 precision;
|
|
stbsp__int32 tz;
|
|
stbsp__uint32 flags;
|
|
|
|
// macros for the callback buffer stuff
|
|
#define stbsp__chk_cb_bufL(bytes) \
|
|
{ \
|
|
int len = (int)(bf - buf); \
|
|
if ((len + (bytes)) >= STB_SPRINTF_MIN){ \
|
|
tlen += len; \
|
|
if (0 == (bf = buf = callback(buf, user, len))) goto done; \
|
|
} \
|
|
}
|
|
|
|
#define stbsp__chk_cb_buf(bytes) \
|
|
{ \
|
|
if (callback) { \
|
|
stbsp__chk_cb_bufL(bytes); \
|
|
} \
|
|
}
|
|
|
|
#define stbsp__flush_cb() \
|
|
{ \
|
|
stbsp__chk_cb_bufL(STB_SPRINTF_MIN - 1); \
|
|
} // flush if there is even one byte in the buffer
|
|
|
|
#define stbsp__cb_buf_clamp(cl, v) \
|
|
cl = v; \
|
|
if (callback) { \
|
|
int lg = STB_SPRINTF_MIN - (int)(bf - buf); \
|
|
if (cl > lg) \
|
|
cl = lg; \
|
|
}
|
|
|
|
// fast copy everything up to the next % (or end of string)
|
|
for (;;) {
|
|
while (((stbsp__uintptr)f) & 3) {
|
|
schk1:
|
|
if (f[0] == '%') goto scandd;
|
|
schk2:
|
|
if (f[0] == 0) goto endfmt;
|
|
stbsp__chk_cb_buf(1);
|
|
*bf++ = f[0];
|
|
++f;
|
|
}
|
|
for (;;) {
|
|
// Check if the next 4 bytes contain %(0x25) or end of string.
|
|
// Using the 'hasless' trick:
|
|
// https://graphics.stanford.edu/~seander/bithacks.html#HasLessInWord
|
|
stbsp__uint32 v, c;
|
|
v = *(stbsp__uint32 *)f;
|
|
c = (~v) & 0x80808080;
|
|
if (((v ^ 0x25252525) - 0x01010101) & c)
|
|
goto schk1;
|
|
if ((v - 0x01010101) & c)
|
|
goto schk2;
|
|
if (callback)
|
|
if ((STB_SPRINTF_MIN - (int)(bf - buf)) < 4)
|
|
goto schk1;
|
|
#ifdef STB_SPRINTF_NOUNALIGNED
|
|
if(((stbsp__uintptr)bf) & 3) {
|
|
bf[0] = f[0];
|
|
bf[1] = f[1];
|
|
bf[2] = f[2];
|
|
bf[3] = f[3];
|
|
} else
|
|
#endif
|
|
{
|
|
*(stbsp__uint32 *)bf = v;
|
|
}
|
|
bf += 4;
|
|
f += 4;
|
|
}
|
|
}
|
|
scandd:
|
|
|
|
++f;
|
|
|
|
// ok, we have a percent, read the modifiers first
|
|
field_width = 0;
|
|
precision = -1;
|
|
flags = 0;
|
|
tz = 0;
|
|
|
|
// flags
|
|
for (;;) {
|
|
switch (f[0]) {
|
|
// if we have left justify
|
|
case '-':
|
|
flags |= STBSP__LEFTJUST;
|
|
++f;
|
|
continue;
|
|
// if we have leading plus
|
|
case '+':
|
|
flags |= STBSP__LEADINGPLUS;
|
|
++f;
|
|
continue;
|
|
// if we have leading space
|
|
case ' ':
|
|
flags |= STBSP__LEADINGSPACE;
|
|
++f;
|
|
continue;
|
|
// if we have leading 0x
|
|
case '#':
|
|
flags |= STBSP__LEADING_0X;
|
|
++f;
|
|
continue;
|
|
// if we have thousand commas
|
|
case '\'':
|
|
flags |= STBSP__TRIPLET_COMMA;
|
|
++f;
|
|
continue;
|
|
// if we have memory sizes
|
|
case '$':
|
|
flags |= STBSP__MEMORY_SIZES;
|
|
++f;
|
|
continue;
|
|
// if we have leading zero
|
|
case '0':
|
|
flags |= STBSP__LEADINGZERO;
|
|
++f;
|
|
goto flags_done;
|
|
default: goto flags_done;
|
|
}
|
|
}
|
|
flags_done:
|
|
|
|
// get the field width
|
|
if (f[0] == '*'){
|
|
field_width = va_arg(va, stbsp__uint32);
|
|
++f;
|
|
}
|
|
else{
|
|
while ((f[0] >= '0') && (f[0] <= '9')) {
|
|
field_width = field_width*10 + f[0] - '0';
|
|
f++;
|
|
}
|
|
}
|
|
|
|
// get the precision
|
|
if (f[0] == '.'){
|
|
++f;
|
|
if (f[0] == '*'){
|
|
precision = va_arg(va, stbsp__uint32);
|
|
++f;
|
|
}
|
|
else{
|
|
precision = 0;
|
|
while ((f[0] >= '0') && (f[0] <= '9')){
|
|
precision = precision*10 + f[0] - '0';
|
|
f++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// handle integer size overrides
|
|
switch (f[0]){
|
|
// are we halfwidth?
|
|
case 'h':
|
|
flags |= STBSP__HALFWIDTH;
|
|
++f;
|
|
if (f[0] == 'h')
|
|
++f; // QUARTERWIDTH
|
|
break;
|
|
// are we 64-bit (unix style)
|
|
case 'l':
|
|
flags |= ((sizeof(long) == 8) ? STBSP__INTMAX : 0);
|
|
++f;
|
|
if (f[0] == 'l') {
|
|
flags |= STBSP__INTMAX;
|
|
++f;
|
|
}
|
|
break;
|
|
// are we 64-bit on intmax? (c99)
|
|
case 'j':
|
|
flags |= (sizeof(size_t) == 8) ? STBSP__INTMAX : 0;
|
|
++f;
|
|
break;
|
|
// are we 64-bit on size_t or ptrdiff_t? (c99)
|
|
case 'z':
|
|
flags |= (sizeof(ptrdiff_t) == 8) ? STBSP__INTMAX : 0;
|
|
++f;
|
|
break;
|
|
case 't':
|
|
flags |= (sizeof(ptrdiff_t) == 8) ? STBSP__INTMAX : 0;
|
|
++f;
|
|
break;
|
|
// are we 64-bit (msft style)
|
|
case 'I':
|
|
if ((f[1] == '6') && (f[2] == '4')) {
|
|
flags |= STBSP__INTMAX;
|
|
f += 3;
|
|
} else if ((f[1] == '3') && (f[2] == '2')) {
|
|
f += 3;
|
|
} else {
|
|
flags |= ((sizeof(void *) == 8) ? STBSP__INTMAX : 0);
|
|
++f;
|
|
}
|
|
break;
|
|
default: break;
|
|
}
|
|
|
|
// handle each replacement
|
|
switch (f[0]){
|
|
#define STBSP__NUMSZ 512 // big enough for e308 (with commas) or e-307
|
|
char num[STBSP__NUMSZ];
|
|
char lead[8];
|
|
char tail[8];
|
|
char *s;
|
|
char const *h;
|
|
stbsp__uint32 l;
|
|
stbsp__uint32 n;
|
|
stbsp__uint32 cs;
|
|
stbsp__uint64 n64;
|
|
double fv;
|
|
stbsp__int32 dp;
|
|
char const *sn;
|
|
|
|
// NOTE(allen): MODIFICATION
|
|
// %S prints str (parameter: String8 str)
|
|
case 'S':
|
|
{
|
|
String8 *str = va_arg(va, String8*);
|
|
s = (char*)str->str;
|
|
l = (precision >= 0 && precision < str->size) ? precision : (U32)str->size;
|
|
lead[0] = 0;
|
|
tail[0] = 0;
|
|
precision = 0;
|
|
dp = 0;
|
|
cs = 0;
|
|
goto scopy;
|
|
}
|
|
|
|
// NOTE(allen): MODIFICATION
|
|
// %N prints n spaces (parameter: int n)
|
|
case 'N':
|
|
{
|
|
// TODO(allen): there's probably a better way to set this
|
|
// up so that it doesn't rely on the buffer of spaces -
|
|
// perhaps a way to apply the space filling in the left justify path?
|
|
l = va_arg(va, int);
|
|
l = (sizeof(spaces) - 1 < l) ? (sizeof(spaces) - 1) : l;
|
|
l = (precision >= 0 && precision < l) ? precision : l;
|
|
s = spaces;
|
|
lead[0] = 0;
|
|
tail[0] = 0;
|
|
precision = 0;
|
|
dp = 0;
|
|
cs = 0;
|
|
goto scopy;
|
|
}
|
|
|
|
case 's':
|
|
{
|
|
// get the string
|
|
s = va_arg(va, char *);
|
|
if (s == 0){
|
|
s = (char *)"null";
|
|
}
|
|
// get the length, limited to desired precision
|
|
// always limit to ~0u chars since our counts are 32b
|
|
l = stbsp__strlen_limited(s, (precision >= 0) ? precision : ~0u);
|
|
lead[0] = 0;
|
|
tail[0] = 0;
|
|
precision = 0;
|
|
dp = 0;
|
|
cs = 0;
|
|
// copy the string in
|
|
goto scopy;
|
|
}
|
|
|
|
case 'c': // char
|
|
{
|
|
// get the character
|
|
s = num + STBSP__NUMSZ - 1;
|
|
*s = (char)va_arg(va, int);
|
|
l = 1;
|
|
lead[0] = 0;
|
|
tail[0] = 0;
|
|
precision = 0;
|
|
dp = 0;
|
|
cs = 0;
|
|
goto scopy;
|
|
}
|
|
|
|
case 'n': // weird write-bytes specifier
|
|
{
|
|
int *d = va_arg(va, int *);
|
|
*d = tlen + (int)(bf - buf);
|
|
} break;
|
|
|
|
case 'A': // hex float
|
|
case 'a': // hex float
|
|
{
|
|
h = (f[0] == 'A') ? hexu : hex;
|
|
fv = va_arg(va, double);
|
|
if (precision == -1){
|
|
precision = 6; // default is 6
|
|
}
|
|
// read the double into a string
|
|
if (stbsp__real_to_parts((stbsp__int64 *)&n64, &dp, fv)){
|
|
flags |= STBSP__NEGATIVE;
|
|
}
|
|
|
|
s = num + 64;
|
|
|
|
stbsp__lead_sign(flags, lead);
|
|
|
|
if (dp == -1023){
|
|
dp = (n64) ? -1022 : 0;
|
|
}
|
|
else{
|
|
n64 |= (((stbsp__uint64)1) << 52);
|
|
}
|
|
n64 <<= (64 - 56);
|
|
if (precision < 15){
|
|
n64 += ((((stbsp__uint64)8) << 56) >> (precision*4));
|
|
}
|
|
// add leading chars
|
|
|
|
lead[1 + lead[0]] = '0';
|
|
lead[2 + lead[0]] = 'x';
|
|
lead[0] += 2;
|
|
|
|
*s++ = h[(n64 >> 60) & 15];
|
|
n64 <<= 4;
|
|
if (precision){
|
|
*s++ = stbsp__period;
|
|
}
|
|
sn = s;
|
|
|
|
// print the bits
|
|
n = precision;
|
|
if (n > 13){
|
|
n = 13;
|
|
}
|
|
if (precision > (stbsp__int32)n){
|
|
tz = precision - n;
|
|
}
|
|
precision = 0;
|
|
while (n--) {
|
|
*s++ = h[(n64 >> 60) & 15];
|
|
n64 <<= 4;
|
|
}
|
|
|
|
// print the expo
|
|
tail[1] = h[17];
|
|
if (dp < 0){
|
|
tail[2] = '-';
|
|
dp = -dp;
|
|
}
|
|
else{
|
|
tail[2] = '+';
|
|
}
|
|
n = (dp >= 1000) ? 6 : ((dp >= 100) ? 5 : ((dp >= 10) ? 4 : 3));
|
|
tail[0] = (char)n;
|
|
for (;;) {
|
|
tail[n] = '0' + dp % 10;
|
|
if (n <= 3) break;
|
|
--n;
|
|
dp /= 10;
|
|
}
|
|
|
|
dp = (int)(s - sn);
|
|
l = (int)(s - (num + 64));
|
|
s = num + 64;
|
|
cs = 1 + (3 << 24);
|
|
goto scopy;
|
|
}
|
|
|
|
case 'G': // float
|
|
case 'g': // float
|
|
{
|
|
h = (f[0] == 'G') ? hexu : hex;
|
|
fv = va_arg(va, double);
|
|
if (precision == -1){
|
|
precision = 6;
|
|
}
|
|
else if (precision == 0){
|
|
precision = 1; // default is 6
|
|
}
|
|
// read the double into a string
|
|
if (stbsp__real_to_str(&sn, &l, num, &dp, fv, (precision - 1) | 0x80000000)){
|
|
flags |= STBSP__NEGATIVE;
|
|
}
|
|
|
|
// clamp the precision and delete extra zeros after clamp
|
|
n = precision;
|
|
if (l > (stbsp__uint32)precision){
|
|
l = precision;
|
|
}
|
|
while ((l > 1) && (precision) && (sn[l - 1] == '0')){
|
|
--precision;
|
|
--l;
|
|
}
|
|
|
|
// should we use %e
|
|
if ((dp <= -4) || (dp > (stbsp__int32)n)) {
|
|
if (precision > (stbsp__int32)l){
|
|
precision = l - 1;
|
|
}
|
|
else if (precision){
|
|
--precision; // when using %e, there is one digit before the decimal
|
|
}
|
|
goto doexpfromg;
|
|
}
|
|
// this is the insane action to get the precision to match %g semantics for %f
|
|
if (dp > 0){
|
|
precision = (dp < (stbsp__int32)l) ? l - dp : 0;
|
|
}
|
|
else{
|
|
precision = -dp + ((precision > (stbsp__int32)l) ? (stbsp__int32) l : precision);
|
|
}
|
|
goto dofloatfromg;
|
|
}
|
|
|
|
case 'E': // float
|
|
case 'e': // float
|
|
{
|
|
h = (f[0] == 'E') ? hexu : hex;
|
|
fv = va_arg(va, double);
|
|
if (precision == -1){
|
|
precision = 6; // default is 6
|
|
}
|
|
// read the double into a string
|
|
if (stbsp__real_to_str(&sn, &l, num, &dp, fv, precision | 0x80000000)){
|
|
flags |= STBSP__NEGATIVE;
|
|
}
|
|
goto doexpfromg;
|
|
}
|
|
|
|
doexpfromg:
|
|
{
|
|
tail[0] = 0;
|
|
stbsp__lead_sign(flags, lead);
|
|
if (dp == STBSP__SPECIAL) {
|
|
s = (char *)sn;
|
|
cs = 0;
|
|
precision = 0;
|
|
goto scopy;
|
|
}
|
|
s = num + 64;
|
|
// handle leading chars
|
|
*s++ = sn[0];
|
|
|
|
if (precision){
|
|
*s++ = stbsp__period;
|
|
}
|
|
|
|
// handle after decimal
|
|
if ((l - 1) > (stbsp__uint32)precision){
|
|
l = precision + 1;
|
|
}
|
|
for (n = 1; n < l; n++){
|
|
*s++ = sn[n];
|
|
}
|
|
// trailing zeros
|
|
tz = precision - (l - 1);
|
|
precision = 0;
|
|
// dump expo
|
|
tail[1] = h[0xe];
|
|
dp -= 1;
|
|
if (dp < 0){
|
|
tail[2] = '-';
|
|
dp = -dp;
|
|
}
|
|
else{
|
|
tail[2] = '+';
|
|
}
|
|
n = (dp >= 100) ? 5 : 4;
|
|
tail[0] = (char)n;
|
|
for (;;) {
|
|
tail[n] = '0' + dp % 10;
|
|
if (n <= 3)
|
|
break;
|
|
--n;
|
|
dp /= 10;
|
|
}
|
|
cs = 1 + (3 << 24); // how many tens
|
|
goto flt_lead;
|
|
}
|
|
|
|
case 'f': // float
|
|
{
|
|
fv = va_arg(va, double);
|
|
goto doafloat;
|
|
}
|
|
|
|
doafloat:
|
|
{
|
|
// do kilos
|
|
if (flags & STBSP__MEMORY_SIZES) {
|
|
double divisor = 1024.f;
|
|
while (flags < 0x4000000) {
|
|
if ((fv < divisor) && (fv > -divisor))
|
|
break;
|
|
fv /= divisor;
|
|
flags += 0x1000000;
|
|
}
|
|
}
|
|
if (precision == -1){
|
|
precision = 6; // default is 6
|
|
}
|
|
// read the double into a string
|
|
if (stbsp__real_to_str(&sn, &l, num, &dp, fv, precision)){
|
|
flags |= STBSP__NEGATIVE;
|
|
}
|
|
}
|
|
|
|
dofloatfromg:
|
|
{
|
|
tail[0] = 0;
|
|
stbsp__lead_sign(flags, lead);
|
|
if (dp == STBSP__SPECIAL) {
|
|
s = (char *)sn;
|
|
cs = 0;
|
|
precision = 0;
|
|
goto scopy;
|
|
}
|
|
s = num + 64;
|
|
|
|
// handle the three decimal varieties
|
|
if (dp <= 0) {
|
|
stbsp__int32 i;
|
|
// handle 0.000*000xxxx
|
|
*s++ = '0';
|
|
if (precision){
|
|
*s++ = stbsp__period;
|
|
}
|
|
n = -dp;
|
|
if ((stbsp__int32)n > precision){
|
|
n = precision;
|
|
}
|
|
i = n;
|
|
while (i) {
|
|
if ((((stbsp__uintptr)s) & 3) == 0)
|
|
break;
|
|
*s++ = '0';
|
|
--i;
|
|
}
|
|
while (i >= 4) {
|
|
*(stbsp__uint32 *)s = 0x30303030;
|
|
s += 4;
|
|
i -= 4;
|
|
}
|
|
while (i) {
|
|
*s++ = '0';
|
|
--i;
|
|
}
|
|
if ((stbsp__int32)(l + n) > precision){
|
|
l = precision - n;
|
|
}
|
|
i = l;
|
|
while (i) {
|
|
*s++ = *sn++;
|
|
--i;
|
|
}
|
|
tz = precision - (n + l);
|
|
cs = 1 + (3 << 24); // how many tens did we write (for commas below)
|
|
}
|
|
else{
|
|
cs = (flags & STBSP__TRIPLET_COMMA) ? ((600 - (stbsp__uint32)dp) % 3) : 0;
|
|
if ((stbsp__uint32)dp >= l) {
|
|
// handle xxxx000*000.0
|
|
n = 0;
|
|
for (;;) {
|
|
if ((flags & STBSP__TRIPLET_COMMA) && (++cs == 4)) {
|
|
cs = 0;
|
|
*s++ = stbsp__comma;
|
|
} else {
|
|
*s++ = sn[n];
|
|
++n;
|
|
if (n >= l)
|
|
break;
|
|
}
|
|
}
|
|
if (n < (stbsp__uint32)dp) {
|
|
n = dp - n;
|
|
if ((flags & STBSP__TRIPLET_COMMA) == 0) {
|
|
while (n) {
|
|
if ((((stbsp__uintptr)s) & 3) == 0)
|
|
break;
|
|
*s++ = '0';
|
|
--n;
|
|
}
|
|
while (n >= 4) {
|
|
*(stbsp__uint32 *)s = 0x30303030;
|
|
s += 4;
|
|
n -= 4;
|
|
}
|
|
}
|
|
while (n) {
|
|
if ((flags & STBSP__TRIPLET_COMMA) && (++cs == 4)) {
|
|
cs = 0;
|
|
*s++ = stbsp__comma;
|
|
} else {
|
|
*s++ = '0';
|
|
--n;
|
|
}
|
|
}
|
|
}
|
|
cs = (int)(s - (num + 64)) + (3 << 24); // cs is how many tens
|
|
if (precision) {
|
|
*s++ = stbsp__period;
|
|
tz = precision;
|
|
}
|
|
} else {
|
|
// handle xxxxx.xxxx000*000
|
|
n = 0;
|
|
for (;;) {
|
|
if ((flags & STBSP__TRIPLET_COMMA) && (++cs == 4)) {
|
|
cs = 0;
|
|
*s++ = stbsp__comma;
|
|
} else {
|
|
*s++ = sn[n];
|
|
++n;
|
|
if (n >= (stbsp__uint32)dp)
|
|
break;
|
|
}
|
|
}
|
|
cs = (int)(s - (num + 64)) + (3 << 24); // cs is how many tens
|
|
if (precision){
|
|
*s++ = stbsp__period;
|
|
}
|
|
if ((l - dp) > (stbsp__uint32)precision){
|
|
l = precision + dp;
|
|
}
|
|
while (n < l) {
|
|
*s++ = sn[n];
|
|
++n;
|
|
}
|
|
tz = precision - (l - dp);
|
|
}
|
|
}
|
|
precision = 0;
|
|
|
|
// handle k,m,g,t
|
|
if (flags & STBSP__MEMORY_SIZES) {
|
|
tail[0] = 0;
|
|
{
|
|
char idx = 1;
|
|
tail[idx] = ' ';
|
|
idx++;
|
|
tail[idx] = " KMGT"[flags >> 24];
|
|
idx++;
|
|
tail[idx] = 'b';
|
|
tail[0] = idx;
|
|
}
|
|
}
|
|
goto flt_lead;
|
|
}
|
|
|
|
flt_lead:
|
|
{
|
|
// get the length that we copied
|
|
l = (stbsp__uint32)(s - (num + 64));
|
|
s = num + 64;
|
|
goto scopy;
|
|
}
|
|
|
|
case 'B': // upper binary
|
|
case 'b': // lower binary
|
|
{
|
|
h = (f[0] == 'B') ? hexu : hex;
|
|
lead[0] = 0;
|
|
if (flags & STBSP__LEADING_0X) {
|
|
lead[0] = 2;
|
|
lead[1] = '0';
|
|
lead[2] = h[0xb];
|
|
}
|
|
l = (8 << 4) | (1 << 8);
|
|
goto radixnum;
|
|
}
|
|
|
|
case 'o': // octal
|
|
{
|
|
h = hexu;
|
|
lead[0] = 0;
|
|
if (flags & STBSP__LEADING_0X) {
|
|
lead[0] = 1;
|
|
lead[1] = '0';
|
|
}
|
|
l = (3 << 4) | (3 << 8);
|
|
goto radixnum;
|
|
}
|
|
|
|
case 'p': // pointer
|
|
{
|
|
flags |= (sizeof(void *) == 8) ? STBSP__INTMAX : 0;
|
|
precision = sizeof(void *) * 2;
|
|
flags &= ~STBSP__LEADINGZERO; // 'p' only prints the pointer with zeros
|
|
// fall through - to X
|
|
}
|
|
|
|
case 'X': // upper hex
|
|
case 'x': // lower hex
|
|
{
|
|
h = (f[0] == 'X') ? hexu : hex;
|
|
l = (4 << 4) | (4 << 8);
|
|
lead[0] = 0;
|
|
if (flags & STBSP__LEADING_0X) {
|
|
lead[0] = 2;
|
|
lead[1] = '0';
|
|
lead[2] = h[16];
|
|
}
|
|
goto radixnum;
|
|
}
|
|
|
|
radixnum:
|
|
{
|
|
// get the number
|
|
if (flags & STBSP__INTMAX){
|
|
n64 = va_arg(va, stbsp__uint64);
|
|
}
|
|
else{
|
|
n64 = va_arg(va, stbsp__uint32);
|
|
}
|
|
|
|
s = num + STBSP__NUMSZ;
|
|
dp = 0;
|
|
// clear tail, and clear leading if value is zero
|
|
tail[0] = 0;
|
|
if (n64 == 0) {
|
|
lead[0] = 0;
|
|
if (precision == 0) {
|
|
l = 0;
|
|
cs = 0;
|
|
goto scopy;
|
|
}
|
|
}
|
|
// convert to string
|
|
for (;;) {
|
|
*--s = h[n64 & ((1 << (l >> 8)) - 1)];
|
|
n64 >>= (l >> 8);
|
|
if (!((n64) || ((stbsp__int32)((num + STBSP__NUMSZ) - s) < precision))){
|
|
break;
|
|
}
|
|
if (flags & STBSP__TRIPLET_COMMA) {
|
|
++l;
|
|
if ((l & 15) == ((l >> 4) & 15)) {
|
|
l &= ~15;
|
|
*--s = stbsp__comma;
|
|
}
|
|
}
|
|
};
|
|
// get the tens and the comma pos
|
|
cs = (stbsp__uint32)((num + STBSP__NUMSZ) - s) + ((((l >> 4) & 15)) << 24);
|
|
// get the length that we copied
|
|
l = (stbsp__uint32)((num + STBSP__NUMSZ) - s);
|
|
// copy it
|
|
goto scopy;
|
|
}
|
|
|
|
case 'u': // unsigned
|
|
case 'i':
|
|
case 'd': // integer
|
|
{
|
|
// get the integer and abs it
|
|
if (flags & STBSP__INTMAX) {
|
|
stbsp__int64 i64 = va_arg(va, stbsp__int64);
|
|
n64 = (stbsp__uint64)i64;
|
|
if ((f[0] != 'u') && (i64 < 0)) {
|
|
n64 = (stbsp__uint64)-i64;
|
|
flags |= STBSP__NEGATIVE;
|
|
}
|
|
} else {
|
|
stbsp__int32 i = va_arg(va, stbsp__int32);
|
|
n64 = (stbsp__uint32)i;
|
|
if ((f[0] != 'u') && (i < 0)) {
|
|
n64 = (stbsp__uint32)-i;
|
|
flags |= STBSP__NEGATIVE;
|
|
}
|
|
}
|
|
|
|
if (flags & STBSP__MEMORY_SIZES) {
|
|
if (n64 < 1024){
|
|
precision = 0;
|
|
}
|
|
else if (precision == -1){
|
|
precision = 1;
|
|
}
|
|
fv = (double)(stbsp__int64)n64;
|
|
goto doafloat;
|
|
}
|
|
|
|
// convert to string
|
|
s = num + STBSP__NUMSZ;
|
|
l = 0;
|
|
|
|
for (;;) {
|
|
// do in 32-bit chunks (avoid lots of 64-bit divides even with constant denominators)
|
|
char *o = s - 8;
|
|
if (n64 >= 100000000) {
|
|
n = (stbsp__uint32)(n64 % 100000000);
|
|
n64 /= 100000000;
|
|
} else {
|
|
n = (stbsp__uint32)n64;
|
|
n64 = 0;
|
|
}
|
|
if ((flags & STBSP__TRIPLET_COMMA) == 0) {
|
|
do {
|
|
s -= 2;
|
|
*(stbsp__uint16 *)s = *(stbsp__uint16 *)&stbsp__digitpair.pair[(n % 100) * 2];
|
|
n /= 100;
|
|
} while (n);
|
|
}
|
|
while (n) {
|
|
if ((flags & STBSP__TRIPLET_COMMA) && (l++ == 3)) {
|
|
l = 0;
|
|
*--s = stbsp__comma;
|
|
--o;
|
|
} else {
|
|
*--s = (char)(n % 10) + '0';
|
|
n /= 10;
|
|
}
|
|
}
|
|
if (n64 == 0) {
|
|
if ((s[0] == '0') && (s != (num + STBSP__NUMSZ))){
|
|
++s;
|
|
}
|
|
break;
|
|
}
|
|
while (s != o){
|
|
if ((flags & STBSP__TRIPLET_COMMA) && (l++ == 3)){
|
|
l = 0;
|
|
*--s = stbsp__comma;
|
|
--o;
|
|
}
|
|
else{
|
|
*--s = '0';
|
|
}
|
|
}
|
|
}
|
|
|
|
tail[0] = 0;
|
|
stbsp__lead_sign(flags, lead);
|
|
|
|
// get the length that we copied
|
|
l = (stbsp__uint32)((num + STBSP__NUMSZ) - s);
|
|
if (l == 0) {
|
|
*--s = '0';
|
|
l = 1;
|
|
}
|
|
cs = l + (3 << 24);
|
|
if (precision < 0){
|
|
precision = 0;
|
|
}
|
|
goto scopy;
|
|
}
|
|
|
|
default: // unknown, just copy code
|
|
{
|
|
s = num + STBSP__NUMSZ - 1;
|
|
*s = f[0];
|
|
l = 1;
|
|
field_width = flags = 0;
|
|
lead[0] = 0;
|
|
tail[0] = 0;
|
|
precision = 0;
|
|
dp = 0;
|
|
cs = 0;
|
|
goto scopy;
|
|
}
|
|
|
|
scopy:
|
|
{
|
|
// get field_width=leading/trailing space, precision=leading zeros
|
|
if (precision < (stbsp__int32)l){
|
|
precision = l;
|
|
}
|
|
n = precision + lead[0] + tail[0] + tz;
|
|
if (field_width < (stbsp__int32)n){
|
|
field_width = n;
|
|
}
|
|
field_width -= n;
|
|
precision -= l;
|
|
|
|
// handle right justify and leading zeros
|
|
if ((flags & STBSP__LEFTJUST) == 0){
|
|
// if leading zeros, everything is in precision
|
|
if (flags & STBSP__LEADINGZERO){
|
|
precision = (field_width > precision) ? field_width : precision;
|
|
field_width = 0;
|
|
}
|
|
// if no leading zeros, then no commas
|
|
else{
|
|
flags &= ~STBSP__TRIPLET_COMMA;
|
|
}
|
|
}
|
|
|
|
// copy the spaces and/or zeros
|
|
if (field_width + precision) {
|
|
stbsp__int32 i;
|
|
stbsp__uint32 c;
|
|
|
|
// copy leading spaces (or when doing %8.4d stuff)
|
|
if ((flags & STBSP__LEFTJUST) == 0)
|
|
while (field_width > 0) {
|
|
stbsp__cb_buf_clamp(i, field_width);
|
|
field_width -= i;
|
|
while (i) {
|
|
if ((((stbsp__uintptr)bf) & 3) == 0)
|
|
break;
|
|
*bf++ = ' ';
|
|
--i;
|
|
}
|
|
while (i >= 4) {
|
|
*(stbsp__uint32 *)bf = 0x20202020;
|
|
bf += 4;
|
|
i -= 4;
|
|
}
|
|
while (i) {
|
|
*bf++ = ' ';
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
|
|
// copy leader
|
|
sn = lead + 1;
|
|
while (lead[0]) {
|
|
stbsp__cb_buf_clamp(i, lead[0]);
|
|
lead[0] -= (char)i;
|
|
while (i) {
|
|
*bf++ = *sn++;
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
|
|
// copy leading zeros
|
|
c = cs >> 24;
|
|
cs &= 0xffffff;
|
|
cs = (flags & STBSP__TRIPLET_COMMA) ? ((stbsp__uint32)(c - ((precision + cs) % (c + 1)))) : 0;
|
|
while (precision > 0) {
|
|
stbsp__cb_buf_clamp(i, precision);
|
|
precision -= i;
|
|
if ((flags & STBSP__TRIPLET_COMMA) == 0) {
|
|
while (i) {
|
|
if ((((stbsp__uintptr)bf) & 3) == 0)
|
|
break;
|
|
*bf++ = '0';
|
|
--i;
|
|
}
|
|
while (i >= 4) {
|
|
*(stbsp__uint32 *)bf = 0x30303030;
|
|
bf += 4;
|
|
i -= 4;
|
|
}
|
|
}
|
|
while (i) {
|
|
if ((flags & STBSP__TRIPLET_COMMA) && (cs++ == c)) {
|
|
cs = 0;
|
|
*bf++ = stbsp__comma;
|
|
} else
|
|
*bf++ = '0';
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
}
|
|
|
|
// copy leader if there is still one
|
|
sn = lead + 1;
|
|
while (lead[0]) {
|
|
stbsp__int32 i;
|
|
stbsp__cb_buf_clamp(i, lead[0]);
|
|
lead[0] -= (char)i;
|
|
while (i) {
|
|
*bf++ = *sn++;
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
|
|
// copy the string
|
|
n = l;
|
|
while (n) {
|
|
stbsp__int32 i;
|
|
stbsp__cb_buf_clamp(i, n);
|
|
n -= i;
|
|
STBSP__UNALIGNED(while (i >= 4) {
|
|
*(stbsp__uint32 volatile *)bf = *(stbsp__uint32 volatile *)s;
|
|
bf += 4;
|
|
s += 4;
|
|
i -= 4;
|
|
})
|
|
while (i) {
|
|
*bf++ = *s++;
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
|
|
// copy trailing zeros
|
|
while (tz) {
|
|
stbsp__int32 i;
|
|
stbsp__cb_buf_clamp(i, tz);
|
|
tz -= i;
|
|
while (i) {
|
|
if ((((stbsp__uintptr)bf) & 3) == 0)
|
|
break;
|
|
*bf++ = '0';
|
|
--i;
|
|
}
|
|
while (i >= 4) {
|
|
*(stbsp__uint32 *)bf = 0x30303030;
|
|
bf += 4;
|
|
i -= 4;
|
|
}
|
|
while (i) {
|
|
*bf++ = '0';
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
|
|
// copy tail if there is one
|
|
sn = tail + 1;
|
|
while (tail[0]) {
|
|
stbsp__int32 i;
|
|
stbsp__cb_buf_clamp(i, tail[0]);
|
|
tail[0] -= (char)i;
|
|
while (i) {
|
|
*bf++ = *sn++;
|
|
--i;
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
|
|
// handle the left justify
|
|
if (flags & STBSP__LEFTJUST){
|
|
if (field_width > 0){
|
|
while (field_width){
|
|
stbsp__int32 i;
|
|
stbsp__cb_buf_clamp(i, field_width);
|
|
field_width -= i;
|
|
while (i){
|
|
if ((((stbsp__uintptr)bf) & 3) == 0)
|
|
break;
|
|
*bf++ = ' ';
|
|
--i;
|
|
}
|
|
while (i >= 4){
|
|
*(stbsp__uint32 *)bf = 0x20202020;
|
|
bf += 4;
|
|
i -= 4;
|
|
}
|
|
while (i--){
|
|
*bf++ = ' ';
|
|
}
|
|
stbsp__chk_cb_buf(1);
|
|
}
|
|
}
|
|
}
|
|
}break;
|
|
}
|
|
++f;
|
|
}
|
|
endfmt:
|
|
|
|
if (!callback)
|
|
*bf = 0;
|
|
else
|
|
stbsp__flush_cb();
|
|
|
|
done:
|
|
return tlen + (int)(bf - buf);
|
|
}
|
|
|
|
// cleanup
|
|
#undef STBSP__LEFTJUST
|
|
#undef STBSP__LEADINGPLUS
|
|
#undef STBSP__LEADINGSPACE
|
|
#undef STBSP__LEADING_0X
|
|
#undef STBSP__LEADINGZERO
|
|
#undef STBSP__INTMAX
|
|
#undef STBSP__TRIPLET_COMMA
|
|
#undef STBSP__NEGATIVE
|
|
#undef STBSP__METRIC_SUFFIX
|
|
#undef STBSP__NUMSZ
|
|
#undef stbsp__chk_cb_bufL
|
|
#undef stbsp__chk_cb_buf
|
|
#undef stbsp__flush_cb
|
|
#undef stbsp__cb_buf_clamp
|
|
|
|
// ============================================================================
|
|
// wrapper functions
|
|
|
|
STBSP__PUBLICDEF int m4_sprintf(char *buf, char const *fmt, ...)
|
|
{
|
|
int result;
|
|
va_list va;
|
|
va_start(va, fmt);
|
|
result = m4_vsprintfcb(0, 0, buf, fmt, va);
|
|
va_end(va);
|
|
return result;
|
|
}
|
|
|
|
typedef struct stbsp__context {
|
|
char *buf;
|
|
int count;
|
|
int length;
|
|
char tmp[STB_SPRINTF_MIN];
|
|
} stbsp__context;
|
|
|
|
static char *stbsp__clamp_callback(const char *buf, void *user, int len)
|
|
{
|
|
stbsp__context *c = (stbsp__context *)user;
|
|
c->length += len;
|
|
|
|
if (len > c->count)
|
|
len = c->count;
|
|
|
|
if (len) {
|
|
if (buf != c->buf) {
|
|
const char *s, *se;
|
|
char *d;
|
|
d = c->buf;
|
|
s = buf;
|
|
se = buf + len;
|
|
do {
|
|
*d++ = *s++;
|
|
} while (s < se);
|
|
}
|
|
c->buf += len;
|
|
c->count -= len;
|
|
}
|
|
|
|
if (c->count <= 0)
|
|
return c->tmp;
|
|
return (c->count >= STB_SPRINTF_MIN) ? c->buf : c->tmp; // go direct into buffer if you can
|
|
}
|
|
|
|
static char * stbsp__count_clamp_callback( const char * buf, void * user, int len )
|
|
{
|
|
stbsp__context * c = (stbsp__context*)user;
|
|
(void) sizeof(buf);
|
|
|
|
c->length += len;
|
|
return c->tmp; // go direct into buffer if you can
|
|
}
|
|
|
|
STBSP__PUBLICDEF int m4_vsnprintf( char * buf, int count, char const * fmt, va_list va )
|
|
{
|
|
stbsp__context c;
|
|
|
|
if ( (count == 0) && !buf )
|
|
{
|
|
c.length = 0;
|
|
|
|
m4_vsprintfcb( stbsp__count_clamp_callback, &c, c.tmp, fmt, va );
|
|
}
|
|
else
|
|
{
|
|
int l;
|
|
|
|
c.buf = buf;
|
|
c.count = count;
|
|
c.length = 0;
|
|
|
|
m4_vsprintfcb( stbsp__clamp_callback, &c, stbsp__clamp_callback(0,&c,0), fmt, va );
|
|
|
|
// zero-terminate
|
|
l = (int)( c.buf - buf );
|
|
if ( l >= count ) // should never be greater, only equal (or less) than count
|
|
l = count - 1;
|
|
buf[l] = 0;
|
|
}
|
|
|
|
return c.length;
|
|
}
|
|
|
|
STBSP__PUBLICDEF int m4_snprintf(char *buf, int count, char const *fmt, ...)
|
|
{
|
|
int result;
|
|
va_list va;
|
|
va_start(va, fmt);
|
|
|
|
result = m4_vsnprintf(buf, count, fmt, va);
|
|
va_end(va);
|
|
|
|
return result;
|
|
}
|
|
|
|
STBSP__PUBLICDEF int m4_vsprintf(char *buf, char const *fmt, va_list va)
|
|
{
|
|
return m4_vsprintfcb(0, 0, buf, fmt, va);
|
|
}
|
|
|
|
// =======================================================================
|
|
// low level float utility functions
|
|
|
|
// copies d to bits w/ strict aliasing (this compiles to nothing on /Ox)
|
|
#define STBSP__COPYFP(dest, src) \
|
|
{ \
|
|
int cn; \
|
|
for (cn = 0; cn < 8; cn++) \
|
|
((char *)&dest)[cn] = ((char *)&src)[cn]; \
|
|
}
|
|
|
|
// get float info
|
|
static stbsp__int32 stbsp__real_to_parts(stbsp__int64 *bits, stbsp__int32 *expo, double value)
|
|
{
|
|
double d;
|
|
stbsp__int64 b = 0;
|
|
|
|
// load value and round at the frac_digits
|
|
d = value;
|
|
|
|
STBSP__COPYFP(b, d);
|
|
|
|
*bits = b & ((((stbsp__uint64)1) << 52) - 1);
|
|
*expo = (stbsp__int32)(((b >> 52) & 2047) - 1023);
|
|
|
|
return (stbsp__int32)((stbsp__uint64) b >> 63);
|
|
}
|
|
|
|
static double const stbsp__bot[23] = {
|
|
1e+000, 1e+001, 1e+002, 1e+003, 1e+004, 1e+005, 1e+006, 1e+007, 1e+008, 1e+009, 1e+010, 1e+011,
|
|
1e+012, 1e+013, 1e+014, 1e+015, 1e+016, 1e+017, 1e+018, 1e+019, 1e+020, 1e+021, 1e+022
|
|
};
|
|
static double const stbsp__negbot[22] = {
|
|
1e-001, 1e-002, 1e-003, 1e-004, 1e-005, 1e-006, 1e-007, 1e-008, 1e-009, 1e-010, 1e-011,
|
|
1e-012, 1e-013, 1e-014, 1e-015, 1e-016, 1e-017, 1e-018, 1e-019, 1e-020, 1e-021, 1e-022
|
|
};
|
|
static double const stbsp__negboterr[22] = {
|
|
-5.551115123125783e-018, -2.0816681711721684e-019, -2.0816681711721686e-020, -4.7921736023859299e-021, -8.1803053914031305e-022, 4.5251888174113741e-023,
|
|
4.5251888174113739e-024, -2.0922560830128471e-025, -6.2281591457779853e-026, -3.6432197315497743e-027, 6.0503030718060191e-028, 2.0113352370744385e-029,
|
|
-3.0373745563400371e-030, 1.1806906454401013e-032, -7.7705399876661076e-032, 2.0902213275965398e-033, -7.1542424054621921e-034, -7.1542424054621926e-035,
|
|
2.4754073164739869e-036, 5.4846728545790429e-037, 9.2462547772103625e-038, -4.8596774326570872e-039
|
|
};
|
|
static double const stbsp__top[13] = {
|
|
1e+023, 1e+046, 1e+069, 1e+092, 1e+115, 1e+138, 1e+161, 1e+184, 1e+207, 1e+230, 1e+253, 1e+276, 1e+299
|
|
};
|
|
static double const stbsp__negtop[13] = {
|
|
1e-023, 1e-046, 1e-069, 1e-092, 1e-115, 1e-138, 1e-161, 1e-184, 1e-207, 1e-230, 1e-253, 1e-276, 1e-299
|
|
};
|
|
static double const stbsp__toperr[13] = {
|
|
8388608,
|
|
6.8601809640529717e+028,
|
|
-7.253143638152921e+052,
|
|
-4.3377296974619174e+075,
|
|
-1.5559416129466825e+098,
|
|
-3.2841562489204913e+121,
|
|
-3.7745893248228135e+144,
|
|
-1.7356668416969134e+167,
|
|
-3.8893577551088374e+190,
|
|
-9.9566444326005119e+213,
|
|
6.3641293062232429e+236,
|
|
-5.2069140800249813e+259,
|
|
-5.2504760255204387e+282
|
|
};
|
|
static double const stbsp__negtoperr[13] = {
|
|
3.9565301985100693e-040, -2.299904345391321e-063, 3.6506201437945798e-086, 1.1875228833981544e-109,
|
|
-5.0644902316928607e-132, -6.7156837247865426e-155, -2.812077463003139e-178, -5.7778912386589953e-201,
|
|
7.4997100559334532e-224, -4.6439668915134491e-247, -6.3691100762962136e-270, -9.436808465446358e-293,
|
|
8.0970921678014997e-317
|
|
};
|
|
|
|
#if defined(_MSC_VER) && (_MSC_VER <= 1200)
|
|
static stbsp__uint64 const stbsp__powten[20] = {
|
|
1,
|
|
10,
|
|
100,
|
|
1000,
|
|
10000,
|
|
100000,
|
|
1000000,
|
|
10000000,
|
|
100000000,
|
|
1000000000,
|
|
10000000000,
|
|
100000000000,
|
|
1000000000000,
|
|
10000000000000,
|
|
100000000000000,
|
|
1000000000000000,
|
|
10000000000000000,
|
|
100000000000000000,
|
|
1000000000000000000,
|
|
10000000000000000000U
|
|
};
|
|
#define stbsp__tento19th ((stbsp__uint64)1000000000000000000)
|
|
#else
|
|
static stbsp__uint64 const stbsp__powten[20] = {
|
|
1,
|
|
10,
|
|
100,
|
|
1000,
|
|
10000,
|
|
100000,
|
|
1000000,
|
|
10000000,
|
|
100000000,
|
|
1000000000,
|
|
10000000000ULL,
|
|
100000000000ULL,
|
|
1000000000000ULL,
|
|
10000000000000ULL,
|
|
100000000000000ULL,
|
|
1000000000000000ULL,
|
|
10000000000000000ULL,
|
|
100000000000000000ULL,
|
|
1000000000000000000ULL,
|
|
10000000000000000000ULL
|
|
};
|
|
#define stbsp__tento19th (1000000000000000000ULL)
|
|
#endif
|
|
|
|
#define stbsp__ddmulthi(oh, ol, xh, yh) \
|
|
{ \
|
|
double ahi = 0, alo, bhi = 0, blo; \
|
|
stbsp__int64 bt; \
|
|
oh = xh * yh; \
|
|
STBSP__COPYFP(bt, xh); \
|
|
bt &= ((~(stbsp__uint64)0) << 27); \
|
|
STBSP__COPYFP(ahi, bt); \
|
|
alo = xh - ahi; \
|
|
STBSP__COPYFP(bt, yh); \
|
|
bt &= ((~(stbsp__uint64)0) << 27); \
|
|
STBSP__COPYFP(bhi, bt); \
|
|
blo = yh - bhi; \
|
|
ol = ((ahi * bhi - oh) + ahi * blo + alo * bhi) + alo * blo; \
|
|
}
|
|
|
|
#define stbsp__ddtoS64(ob, xh, xl) \
|
|
{ \
|
|
double ahi = 0, alo, vh, t; \
|
|
ob = (stbsp__int64)xh; \
|
|
vh = (double)ob; \
|
|
ahi = (xh - vh); \
|
|
t = (ahi - xh); \
|
|
alo = (xh - (ahi - t)) - (vh + t); \
|
|
ob += (stbsp__int64)(ahi + alo + xl); \
|
|
}
|
|
|
|
#define stbsp__ddrenorm(oh, ol) \
|
|
{ \
|
|
double s; \
|
|
s = oh + ol; \
|
|
ol = ol - (s - oh); \
|
|
oh = s; \
|
|
}
|
|
|
|
#define stbsp__ddmultlo(oh, ol, xh, xl, yh, yl) ol = ol + (xh * yl + xl * yh);
|
|
|
|
#define stbsp__ddmultlos(oh, ol, xh, yl) ol = ol + (xh * yl);
|
|
|
|
static void stbsp__raise_to_power10(double *ohi, double *olo, double d, stbsp__int32 power) // power can be -323 to +350
|
|
{
|
|
double ph, pl;
|
|
if ((power >= 0) && (power <= 22)) {
|
|
stbsp__ddmulthi(ph, pl, d, stbsp__bot[power]);
|
|
} else {
|
|
stbsp__int32 e, et, eb;
|
|
double p2h, p2l;
|
|
|
|
e = power;
|
|
if (power < 0)
|
|
e = -e;
|
|
et = (e * 0x2c9) >> 14; /* %23 */
|
|
if (et > 13)
|
|
et = 13;
|
|
eb = e - (et * 23);
|
|
|
|
ph = d;
|
|
pl = 0.0;
|
|
if (power < 0) {
|
|
if (eb) {
|
|
--eb;
|
|
stbsp__ddmulthi(ph, pl, d, stbsp__negbot[eb]);
|
|
stbsp__ddmultlos(ph, pl, d, stbsp__negboterr[eb]);
|
|
}
|
|
if (et) {
|
|
stbsp__ddrenorm(ph, pl);
|
|
--et;
|
|
stbsp__ddmulthi(p2h, p2l, ph, stbsp__negtop[et]);
|
|
stbsp__ddmultlo(p2h, p2l, ph, pl, stbsp__negtop[et], stbsp__negtoperr[et]);
|
|
ph = p2h;
|
|
pl = p2l;
|
|
}
|
|
} else {
|
|
if (eb) {
|
|
e = eb;
|
|
if (eb > 22)
|
|
eb = 22;
|
|
e -= eb;
|
|
stbsp__ddmulthi(ph, pl, d, stbsp__bot[eb]);
|
|
if (e) {
|
|
stbsp__ddrenorm(ph, pl);
|
|
stbsp__ddmulthi(p2h, p2l, ph, stbsp__bot[e]);
|
|
stbsp__ddmultlos(p2h, p2l, stbsp__bot[e], pl);
|
|
ph = p2h;
|
|
pl = p2l;
|
|
}
|
|
}
|
|
if (et) {
|
|
stbsp__ddrenorm(ph, pl);
|
|
--et;
|
|
stbsp__ddmulthi(p2h, p2l, ph, stbsp__top[et]);
|
|
stbsp__ddmultlo(p2h, p2l, ph, pl, stbsp__top[et], stbsp__toperr[et]);
|
|
ph = p2h;
|
|
pl = p2l;
|
|
}
|
|
}
|
|
}
|
|
stbsp__ddrenorm(ph, pl);
|
|
*ohi = ph;
|
|
*olo = pl;
|
|
}
|
|
|
|
// given a float value, returns the significant bits in bits, and the position of the
|
|
// decimal point in decimal_pos. +/-INF and NAN are specified by special values
|
|
// returned in the decimal_pos parameter.
|
|
// frac_digits is absolute normally, but if you want from first significant digits (got %g and %e), or in 0x80000000
|
|
static stbsp__int32 stbsp__real_to_str(char const **start, stbsp__uint32 *len, char *out, stbsp__int32 *decimal_pos, double value, stbsp__uint32 frac_digits)
|
|
{
|
|
double d;
|
|
stbsp__int64 bits = 0;
|
|
stbsp__int32 expo, e, ng, tens;
|
|
|
|
d = value;
|
|
STBSP__COPYFP(bits, d);
|
|
expo = (stbsp__int32)((bits >> 52) & 2047);
|
|
ng = (stbsp__int32)((stbsp__uint64) bits >> 63);
|
|
if (ng)
|
|
d = -d;
|
|
|
|
if (expo == 2047) // is nan or inf?
|
|
{
|
|
*start = (bits & ((((stbsp__uint64)1) << 52) - 1)) ? "NaN" : "Inf";
|
|
*decimal_pos = STBSP__SPECIAL;
|
|
*len = 3;
|
|
return ng;
|
|
}
|
|
|
|
if (expo == 0) // is zero or denormal
|
|
{
|
|
if (((stbsp__uint64) bits << 1) == 0) // do zero
|
|
{
|
|
*decimal_pos = 1;
|
|
*start = out;
|
|
out[0] = '0';
|
|
*len = 1;
|
|
return ng;
|
|
}
|
|
// find the right expo for denormals
|
|
{
|
|
stbsp__int64 v = ((stbsp__uint64)1) << 51;
|
|
while ((bits & v) == 0) {
|
|
--expo;
|
|
v >>= 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
// find the decimal exponent as well as the decimal bits of the value
|
|
{
|
|
double ph, pl;
|
|
|
|
// log10 estimate - very specifically tweaked to hit or undershoot by no more than 1 of log10 of all expos 1..2046
|
|
tens = expo - 1023;
|
|
tens = (tens < 0) ? ((tens * 617) / 2048) : (((tens * 1233) / 4096) + 1);
|
|
|
|
// move the significant bits into position and stick them into an int
|
|
stbsp__raise_to_power10(&ph, &pl, d, 18 - tens);
|
|
|
|
// get full as much precision from double-double as possible
|
|
stbsp__ddtoS64(bits, ph, pl);
|
|
|
|
// check if we undershot
|
|
if (((stbsp__uint64)bits) >= stbsp__tento19th)
|
|
++tens;
|
|
}
|
|
|
|
// now do the rounding in integer land
|
|
frac_digits = (frac_digits & 0x80000000) ? ((frac_digits & 0x7ffffff) + 1) : (tens + frac_digits);
|
|
if ((frac_digits < 24)) {
|
|
stbsp__uint32 dg = 1;
|
|
if ((stbsp__uint64)bits >= stbsp__powten[9])
|
|
dg = 10;
|
|
while ((stbsp__uint64)bits >= stbsp__powten[dg]) {
|
|
++dg;
|
|
if (dg == 20)
|
|
goto noround;
|
|
}
|
|
if (frac_digits < dg) {
|
|
stbsp__uint64 r;
|
|
// add 0.5 at the right position and round
|
|
e = dg - frac_digits;
|
|
if ((stbsp__uint32)e >= 24)
|
|
goto noround;
|
|
r = stbsp__powten[e];
|
|
bits = bits + (r / 2);
|
|
if ((stbsp__uint64)bits >= stbsp__powten[dg])
|
|
++tens;
|
|
bits /= r;
|
|
}
|
|
noround:;
|
|
}
|
|
|
|
// kill long trailing runs of zeros
|
|
if (bits) {
|
|
stbsp__uint32 n;
|
|
for (;;) {
|
|
if (bits <= 0xffffffff)
|
|
break;
|
|
if (bits % 1000)
|
|
goto donez;
|
|
bits /= 1000;
|
|
}
|
|
n = (stbsp__uint32)bits;
|
|
while ((n % 1000) == 0)
|
|
n /= 1000;
|
|
bits = n;
|
|
donez:;
|
|
}
|
|
|
|
// convert to string
|
|
out += 64;
|
|
e = 0;
|
|
for (;;) {
|
|
stbsp__uint32 n;
|
|
char *o = out - 8;
|
|
// do the conversion in chunks of U32s (avoid most 64-bit divides, worth it, constant denomiators be damned)
|
|
if (bits >= 100000000) {
|
|
n = (stbsp__uint32)(bits % 100000000);
|
|
bits /= 100000000;
|
|
} else {
|
|
n = (stbsp__uint32)bits;
|
|
bits = 0;
|
|
}
|
|
while (n) {
|
|
out -= 2;
|
|
*(stbsp__uint16 *)out = *(stbsp__uint16 *)&stbsp__digitpair.pair[(n % 100) * 2];
|
|
n /= 100;
|
|
e += 2;
|
|
}
|
|
if (bits == 0) {
|
|
if ((e) && (out[0] == '0')) {
|
|
++out;
|
|
--e;
|
|
}
|
|
break;
|
|
}
|
|
while (out != o) {
|
|
*--out = '0';
|
|
++e;
|
|
}
|
|
}
|
|
|
|
*decimal_pos = tens;
|
|
*start = out;
|
|
*len = e;
|
|
return ng;
|
|
}
|
|
|
|
#undef stbsp__ddmulthi
|
|
#undef stbsp__ddrenorm
|
|
#undef stbsp__ddmultlo
|
|
#undef stbsp__ddmultlos
|
|
#undef STBSP__SPECIAL
|
|
#undef STBSP__COPYFP
|
|
|
|
// clean up
|
|
#undef stbsp__uint16
|
|
#undef stbsp__uint32
|
|
#undef stbsp__int32
|
|
#undef stbsp__uint64
|
|
#undef stbsp__int64
|
|
#undef STBSP__UNALIGNED
|
|
|