Files
sx/examples/1221-ffi-cstring-returns.sx
agra 731fb8de64 refactor(ffi-linkage): Phase 7.1 — migrate incidental 12xx ffi examples #foreign→extern
12 plain-C examples that use #foreign incidentally (as FFI plumbing, output
unchanged): 1200/1206/1209-1215/1220/1221/1222. Blanket keyword swap; all fn/global
markers (no class forms in 12xx). Empty snapshot diff; corpus validates directly
(all marker'd). Suite green (647 corpus / 444 unit, 0 failed).

KEPT on #foreign (deferred to Phase 8 cutover): identity-#foreign feature tests
(filename ffi-foreign-*: 1205/1207/1216/1218/1219), the equivalence test 1228, and
the diagnostics that assert on #foreign source/message (1172/1174/1620). Comment-only
provenance prose (1223/1229/1230/1231) left intact per Decision-6-recommended.
2026-06-15 06:49:36 +03:00

32 lines
1.2 KiB
Plaintext

// Foreign `-> [:0]u8` / `-> ?[:0]u8` returns: C hands back ONE `char *`;
// the fat sx string is synthesized at the call boundary ({ptr, strlen};
// NULL maps to the optional's null / an empty string) — issue 0128.
// Pre-fix, the call read the pointer register pair as {ptr, len} and the
// length was garbage (bus error on print).
#import "modules/std.sx";
libc :: #library "c";
err_text :: (code: i32) -> [:0]u8 extern libc "strerror";
sig_text :: (sig: i32) -> ?[:0]u8 extern libc "strsignal";
dlerror :: () -> ?[:0]u8 extern libc;
main :: () -> i32 {
// plain: strerror(0) = "Undefined error: 0" on macOS — assert shape,
// not the exact text (locale/platform variance)
t := err_text(2);
if t.len < 5 { print("BUG: strerror too short ({})\n", t.len); return 1; }
print("strerror(2) len ok\n");
// optional, non-null branch
o := sig_text(2);
if o == null { print("BUG: strsignal null\n"); return 2; }
if o!.len < 3 { print("BUG: strsignal too short\n"); return 3; }
print("optional non-null ok\n");
// optional, NULL branch: dlerror() with no pending error is NULL
d := dlerror();
if d != null { print("BUG: dlerror non-null\n"); return 4; }
print("optional null ok\n");
return 0;
}