Skip to content

Commit 1fd61e0

Browse files
committed
Ensure Cstrings are null terminated
Should fix #16499. Two things I'm not sure about: 1. whether it is okay to use `unsafe_load` in this manner? 2. is this the appropriate place to call `String`, or should it be done in `cconvert`?
1 parent deae2df commit 1fd61e0

File tree

2 files changed

+12
-0
lines changed

2 files changed

+12
-0
lines changed

base/c.jl

+5
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,11 @@ function unsafe_convert(::Type{Cstring}, s::String)
8686
if containsnul(p, sizeof(s))
8787
throw(ArgumentError("embedded NULs are not allowed in C strings: $(repr(s))"))
8888
end
89+
if unsafe_load(p, sizeof(s)+1) != 0
90+
# ensure null terminated, copying if necessary
91+
s = String(s.data)
92+
p = unsafe_convert(Ptr{Cchar}, s)
93+
end
8994
return Cstring(p)
9095
end
9196

test/strings/basic.jl

+7
Original file line numberDiff line numberDiff line change
@@ -248,6 +248,13 @@ let s = "ba\0d"
248248
@test_throws ArgumentError Base.unsafe_convert(Cstring, s)
249249
@test_throws ArgumentError Base.unsafe_convert(Cwstring, wstring(s))
250250
end
251+
# issue 16499: ensure Cstrings are NUL terminated
252+
let s = "aaaa"
253+
a = pointer_to_string(pointer(s.data),2)
254+
@test sizeof(a) == 2
255+
b = pointer_to_string(Base.unsafe_convert(Cstring,a))
256+
@test sizeof(b) == 2
257+
end
251258

252259
cstrdup(s) = @static is_windows() ? ccall(:_strdup, Cstring, (Cstring,), s) : ccall(:strdup, Cstring, (Cstring,), s)
253260
let p = cstrdup("hello")

0 commit comments

Comments
 (0)