zig/lib/c/stdlib.zig
GasInfinity 514f6e589c feat(libzigc): use common integer ato* and strto* implementations
* also removes their musl implementation
2026-01-12 00:55:11 +01:00

391 lines
16 KiB
Zig

const std = @import("std");
const common = @import("common.zig");
const builtin = @import("builtin");
const assert = std.debug.assert;
const div_t = std.c.div_t;
const ldiv_t = std.c.ldiv_t;
const lldiv_t = std.c.lldiv_t;
comptime {
if (builtin.target.isMuslLibC() or builtin.target.isWasiLibC()) {
// Functions specific to musl and wasi-libc.
@export(&abs, .{ .name = "abs", .linkage = common.linkage, .visibility = common.visibility });
@export(&labs, .{ .name = "labs", .linkage = common.linkage, .visibility = common.visibility });
@export(&llabs, .{ .name = "llabs", .linkage = common.linkage, .visibility = common.visibility });
@export(&div, .{ .name = "div", .linkage = common.linkage, .visibility = common.visibility });
@export(&ldiv, .{ .name = "ldiv", .linkage = common.linkage, .visibility = common.visibility });
@export(&lldiv, .{ .name = "lldiv", .linkage = common.linkage, .visibility = common.visibility });
@export(&atoi, .{ .name = "atoi", .linkage = common.linkage, .visibility = common.visibility });
@export(&atol, .{ .name = "atol", .linkage = common.linkage, .visibility = common.visibility });
@export(&atoll, .{ .name = "atoll", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtol, .{ .name = "strtol", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoll, .{ .name = "strtoll", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoul, .{ .name = "strtoul", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoull, .{ .name = "strtoull", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoimax, .{ .name = "strtoimax", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoumax, .{ .name = "strtoumax", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtol, .{ .name = "__strtol_internal", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoll, .{ .name = "__strtoll_internal", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoul, .{ .name = "__strtoul_internal", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoull, .{ .name = "__strtoull_internal", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoimax, .{ .name = "__strtoimax_internal", .linkage = common.linkage, .visibility = common.visibility });
@export(&strtoumax, .{ .name = "__strtoumax_internal", .linkage = common.linkage, .visibility = common.visibility });
@export(&qsort_r, .{ .name = "qsort_r", .linkage = common.linkage, .visibility = common.visibility });
@export(&qsort, .{ .name = "qsort", .linkage = common.linkage, .visibility = common.visibility });
@export(&bsearch, .{ .name = "bsearch", .linkage = common.linkage, .visibility = common.visibility });
}
}
fn abs(a: c_int) callconv(.c) c_int {
return @intCast(@abs(a));
}
fn labs(a: c_long) callconv(.c) c_long {
return @intCast(@abs(a));
}
fn llabs(a: c_longlong) callconv(.c) c_longlong {
return @intCast(@abs(a));
}
fn div(a: c_int, b: c_int) callconv(.c) div_t {
return .{
.quot = @divTrunc(a, b),
.rem = @rem(a, b),
};
}
fn ldiv(a: c_long, b: c_long) callconv(.c) ldiv_t {
return .{
.quot = @divTrunc(a, b),
.rem = @rem(a, b),
};
}
fn lldiv(a: c_longlong, b: c_longlong) callconv(.c) lldiv_t {
return .{
.quot = @divTrunc(a, b),
.rem = @rem(a, b),
};
}
fn atoi(str: [*:0]const c_char) callconv(.c) c_int {
return asciiToInteger(c_int, @ptrCast(str));
}
fn atol(str: [*:0]const c_char) callconv(.c) c_long {
return asciiToInteger(c_long, @ptrCast(str));
}
fn atoll(str: [*:0]const c_char) callconv(.c) c_longlong {
return asciiToInteger(c_longlong, @ptrCast(str));
}
fn asciiToInteger(comptime T: type, buf: [*:0]const u8) T {
comptime assert(std.math.isPowerOfTwo(@bitSizeOf(T)));
var current = buf;
while (std.ascii.isWhitespace(current[0])) : (current += 1) {}
// The behaviour *is* undefined if the result cannot be represented
// but as they are usually called with untrusted input we can just handle overflow gracefully.
if (current[0] == '-') return parseDigitsWithSignGenericCharacter(T, u8, current + 1, null, 10, .neg) catch std.math.minInt(T);
if (current[0] == '+') current += 1;
return parseDigitsWithSignGenericCharacter(T, u8, current, null, 10, .pos) catch std.math.maxInt(T);
}
fn strtol(noalias str: [*:0]const c_char, noalias str_end: ?*[*:0]const c_char, base: c_int) callconv(.c) c_long {
return stringToInteger(c_long, @ptrCast(str), if (str_end) |end| @ptrCast(end) else null, base);
}
fn strtoll(noalias str: [*:0]const c_char, noalias str_end: ?*[*:0]const c_char, base: c_int) callconv(.c) c_longlong {
return stringToInteger(c_longlong, @ptrCast(str), if (str_end) |end| @ptrCast(end) else null, base);
}
fn strtoul(noalias str: [*:0]const c_char, noalias str_end: ?*[*:0]const c_char, base: c_int) callconv(.c) c_ulong {
return stringToInteger(c_ulong, @ptrCast(str), if (str_end) |end| @ptrCast(end) else null, base);
}
fn strtoull(noalias str: [*:0]const c_char, noalias str_end: ?*[*:0]const c_char, base: c_int) callconv(.c) c_ulonglong {
return stringToInteger(c_ulonglong, @ptrCast(str), if (str_end) |end| @ptrCast(end) else null, base);
}
// XXX: These belong in inttypes.zig but we'd have to make stringToInteger pub or move it somewhere else.
fn strtoimax(noalias str: [*:0]const c_char, noalias str_end: ?*[*:0]const c_char, base: c_int) callconv(.c) std.c.intmax_t {
return stringToInteger(std.c.intmax_t, @ptrCast(str), if (str_end) |end| @ptrCast(end) else null, base);
}
fn strtoumax(noalias str: [*:0]const c_char, noalias str_end: ?*[*:0]const c_char, base: c_int) callconv(.c) std.c.uintmax_t {
return stringToInteger(std.c.uintmax_t, @ptrCast(str), if (str_end) |end| @ptrCast(end) else null, base);
}
fn stringToInteger(comptime T: type, noalias buf: [*:0]const u8, noalias maybe_end: ?*[*:0]const u8, base: c_int) T {
comptime assert(std.math.isPowerOfTwo(@bitSizeOf(T)));
if (base == 1 or base > 36) {
if (maybe_end) |end| {
end.* = buf;
}
std.c._errno().* = @intFromEnum(std.c.E.INVAL);
return 0;
}
var current = buf;
while (std.ascii.isWhitespace(current[0])) : (current += 1) {}
const negative: bool = switch (current[0]) {
'-' => blk: {
current += 1;
break :blk true;
},
'+' => blk: {
current += 1;
break :blk false;
},
else => false,
};
// The prefix is allowed iff base == 0 or base == base of the prefix
const real_base: u6 = if (current[0] == '0') blk: {
current += 1;
if ((base == 0 or base == 16) and std.ascii.toLower(current[0]) == 'x' and std.ascii.isHex(current[1])) {
current += 1;
break :blk 16;
}
if ((base == 0 or base == 8) and std.ascii.isDigit(current[0])) {
break :blk 8;
}
break :blk switch (base) {
0 => 10,
else => @intCast(base),
};
} else switch (base) {
0 => 10,
else => @intCast(base),
};
if (@typeInfo(T).int.signedness == .unsigned) {
const result = parseDigitsWithSignGenericCharacter(T, u8, current, maybe_end, real_base, .pos) catch {
std.c._errno().* = @intFromEnum(std.c.E.RANGE);
return std.math.maxInt(T);
};
return if (negative) -%result else result;
}
if (negative) return parseDigitsWithSignGenericCharacter(T, u8, current, maybe_end, real_base, .neg) catch blk: {
std.c._errno().* = @intFromEnum(std.c.E.RANGE);
break :blk std.math.minInt(T);
};
return parseDigitsWithSignGenericCharacter(T, u8, current, maybe_end, real_base, .pos) catch blk: {
std.c._errno().* = @intFromEnum(std.c.E.RANGE);
break :blk std.math.maxInt(T);
};
}
fn parseDigitsWithSignGenericCharacter(
comptime T: type,
comptime Char: type,
noalias buf: [*:0]const Char,
noalias maybe_end: ?*[*:0]const Char,
base: u6,
comptime sign: enum { pos, neg },
) error{Overflow}!T {
assert(base >= 2 and base <= 36);
var current = buf;
defer if (maybe_end) |end| {
end.* = current;
};
const add = switch (sign) {
.pos => std.math.add,
.neg => std.math.sub,
};
var value: T = 0;
while (true) {
const c: u8 = std.math.cast(u8, current[0]) orelse break;
if (!std.ascii.isAlphanumeric(c)) break;
const digit: u6 = @intCast(std.fmt.charToDigit(c, base) catch break);
defer current += 1;
value = try std.math.mul(T, value, base);
value = try add(T, value, digit);
}
return value;
}
// NOTE: Despite its name, `qsort` doesn't have to use quicksort or make any complexity or stability guarantee.
fn qsort_r(base: *anyopaque, n: usize, size: usize, compare: *const fn (a: *const anyopaque, b: *const anyopaque, arg: ?*anyopaque) callconv(.c) c_int, arg: ?*anyopaque) callconv(.c) void {
const Context = struct {
base: [*]u8,
size: usize,
compare: *const fn (a: *const anyopaque, b: *const anyopaque, arg: ?*anyopaque) callconv(.c) c_int,
arg: ?*anyopaque,
pub fn lessThan(ctx: @This(), a: usize, b: usize) bool {
return ctx.compare(&ctx.base[a * ctx.size], &ctx.base[b * ctx.size], ctx.arg) < 0;
}
pub fn swap(ctx: @This(), a: usize, b: usize) void {
const a_bytes: []u8 = ctx.base[a * ctx.size ..][0..ctx.size];
const b_bytes: []u8 = ctx.base[b * ctx.size ..][0..ctx.size];
for (a_bytes, b_bytes) |*ab, *bb| {
const tmp = ab.*;
ab.* = bb.*;
bb.* = tmp;
}
}
};
std.mem.sortUnstableContext(0, n, Context{
.base = @ptrCast(base),
.size = size,
.compare = compare,
.arg = arg,
});
}
fn qsort(base: *anyopaque, n: usize, size: usize, compare: *const fn (a: *const anyopaque, b: *const anyopaque) callconv(.c) c_int) callconv(.c) void {
return qsort_r(base, n, size, (struct {
fn wrap(a: *const anyopaque, b: *const anyopaque, arg: ?*anyopaque) callconv(.c) c_int {
const cmp: *const fn (a: *const anyopaque, b: *const anyopaque) callconv(.c) c_int = @ptrCast(@alignCast(arg.?));
return cmp(a, b);
}
}).wrap, @constCast(compare));
}
// NOTE: Despite its name, `bsearch` doesn't need to be implemented using binary search or make any complexity guarantee.
fn bsearch(key: *const anyopaque, base: *const anyopaque, n: usize, size: usize, compare: *const fn (a: *const anyopaque, b: *const anyopaque) callconv(.c) c_int) callconv(.c) ?*anyopaque {
const base_bytes: [*]const u8 = @ptrCast(base);
var low: usize = 0;
var high: usize = n;
while (low < high) {
// Avoid overflowing in the midpoint calculation
const mid = low + (high - low) / 2;
const elem = &base_bytes[mid * size];
switch (std.math.order(compare(key, elem), 0)) {
.eq => return @constCast(elem),
.gt => low = mid + 1,
.lt => high = mid,
}
}
return null;
}
test abs {
const val: c_int = -10;
try std.testing.expectEqual(10, abs(val));
}
test labs {
const val: c_long = -10;
try std.testing.expectEqual(10, labs(val));
}
test llabs {
const val: c_longlong = -10;
try std.testing.expectEqual(10, llabs(val));
}
test div {
const expected: div_t = .{ .quot = 5, .rem = 5 };
try std.testing.expectEqual(expected, div(55, 10));
}
test ldiv {
const expected: ldiv_t = .{ .quot = -6, .rem = 2 };
try std.testing.expectEqual(expected, ldiv(38, -6));
}
test lldiv {
const expected: lldiv_t = .{ .quot = 1, .rem = 2 };
try std.testing.expectEqual(expected, lldiv(5, 3));
}
test atoi {
try std.testing.expectEqual(0, atoi(@ptrCast("stop42true")));
try std.testing.expectEqual(42, atoi(@ptrCast("42true")));
try std.testing.expectEqual(-1, atoi(@ptrCast("-01")));
try std.testing.expectEqual(1, atoi(@ptrCast("+001")));
try std.testing.expectEqual(100, atoi(@ptrCast(" 100")));
try std.testing.expectEqual(500, atoi(@ptrCast("000000000000500")));
try std.testing.expectEqual(1111, atoi(@ptrCast("0000000000001111_0000")));
try std.testing.expectEqual(0, atoi(@ptrCast("0xAA")));
try std.testing.expectEqual(700, atoi(@ptrCast("700B")));
try std.testing.expectEqual(32453, atoi(@ptrCast("+32453more")));
try std.testing.expectEqual(std.math.maxInt(c_int), atoi(@ptrCast(std.fmt.comptimePrint("{d}", .{std.math.maxInt(c_int)}))));
try std.testing.expectEqual(std.math.minInt(c_int), atoi(@ptrCast(std.fmt.comptimePrint("{d}", .{std.math.minInt(c_int)}))));
}
test atol {
try std.testing.expectEqual(0, atol(@ptrCast("stop42true")));
try std.testing.expectEqual(42, atol(@ptrCast("42true")));
try std.testing.expectEqual(-1, atol(@ptrCast("-01")));
try std.testing.expectEqual(1, atol(@ptrCast("+001")));
try std.testing.expectEqual(100, atol(@ptrCast(" 100")));
try std.testing.expectEqual(500, atol(@ptrCast("000000000000500")));
try std.testing.expectEqual(1111, atol(@ptrCast("0000000000001111_0000")));
try std.testing.expectEqual(0, atol(@ptrCast("0xAA")));
try std.testing.expectEqual(700, atol(@ptrCast("700B")));
try std.testing.expectEqual(32453, atol(@ptrCast("+32453more")));
try std.testing.expectEqual(std.math.maxInt(c_long), atol(@ptrCast(std.fmt.comptimePrint("{d}", .{std.math.maxInt(c_long)}))));
try std.testing.expectEqual(std.math.minInt(c_long), atol(@ptrCast(std.fmt.comptimePrint("{d}", .{std.math.minInt(c_long)}))));
}
test atoll {
try std.testing.expectEqual(0, atoll(@ptrCast("stop42true")));
try std.testing.expectEqual(42, atoll(@ptrCast("42true")));
try std.testing.expectEqual(-1, atoll(@ptrCast("-01")));
try std.testing.expectEqual(1, atoll(@ptrCast("+001")));
try std.testing.expectEqual(100, atoll(@ptrCast(" 100")));
try std.testing.expectEqual(500, atoll(@ptrCast("000000000000500")));
try std.testing.expectEqual(1111, atoll(@ptrCast("0000000000001111_0000")));
try std.testing.expectEqual(0, atoll(@ptrCast("0xAA")));
try std.testing.expectEqual(700, atoll(@ptrCast("700B")));
try std.testing.expectEqual(32453, atoll(@ptrCast(" +32453more")));
try std.testing.expectEqual(std.math.maxInt(c_longlong), atoll(@ptrCast(std.fmt.comptimePrint("{d}", .{std.math.maxInt(c_longlong)}))));
try std.testing.expectEqual(std.math.minInt(c_longlong), atoll(@ptrCast(std.fmt.comptimePrint("{d}", .{std.math.minInt(c_longlong)}))));
}
// FIXME: We cannot test strtol, strtoll, strtoul, etc.. here as it must modify errno and libc is not linked in tests
test bsearch {
const Comparison = struct {
pub fn compare(a: *const anyopaque, b: *const anyopaque) callconv(.c) c_int {
const a_u16: *const u16 = @ptrCast(@alignCast(a));
const b_u16: *const u16 = @ptrCast(@alignCast(b));
return switch (std.math.order(a_u16.*, b_u16.*)) {
.gt => 1,
.eq => 0,
.lt => -1,
};
}
};
const items: []const u16 = &.{ 0, 5, 7, 9, 10, 200, 512, 768 };
try std.testing.expectEqual(@as(?*anyopaque, null), bsearch(&@as(u16, 2000), items.ptr, items.len, @sizeOf(u16), Comparison.compare));
for (items) |*value| {
try std.testing.expectEqual(@as(*const anyopaque, value), bsearch(value, items.ptr, items.len, @sizeOf(u16), Comparison.compare));
}
}