~ chicken-core (chicken-5) e3b88bfb10790af391d933843c02050874fbb066
commit e3b88bfb10790af391d933843c02050874fbb066 Author: felix <felix@call-with-current-continuation.org> AuthorDate: Sat May 22 14:40:39 2010 +0200 Commit: felix <felix@call-with-current-continuation.org> CommitDate: Sat May 22 14:40:39 2010 +0200 ##sys#make-c-string checks for embedded NUL bytes (thanks to Peter Bex) diff --git a/library.scm b/library.scm index 59c39db3..f4fdadde 100644 --- a/library.scm +++ b/library.scm @@ -66,6 +66,7 @@ #define C_close_file(p) (C_fclose((C_FILEPTR)(C_port_file(p))), C_SCHEME_UNDEFINED) #define C_a_f64peek(ptr, c, b, i) C_flonum(ptr, ((double *)C_data_pointer(b))[ C_unfix(i) ]) #define C_fetch_c_strlen(b, i) C_fix(strlen((C_char *)C_block_item(b, C_unfix(i)))) +#define C_asciiz_strlen(str) C_fix(strlen(C_c_string(str))) #define C_peek_c_string(b, i, to, len) (C_memcpy(C_data_pointer(to), (C_char *)C_block_item(b, C_unfix(i)), C_unfix(len)), C_SCHEME_UNDEFINED) #define C_free_mptr(p, i) (C_free((void *)C_block_item(p, C_unfix(i))), C_SCHEME_UNDEFINED) #define C_free_sptr(p, i) (C_free((void *)(((C_char **)C_block_item(p, 0))[ C_unfix(i) ])), C_SCHEME_UNDEFINED) @@ -3737,10 +3738,16 @@ EOF ;; *** '4' is platform dependent! (##core#inline_allocate ("C_a_unsigned_int_to_num" 4) (##sys#slot ptr 0)) ) -(define (##sys#make-c-string str) - (##sys#string-append - str - (string (##core#inline "C_make_character" (##core#inline "C_unfix" 0)))) ) +(define (##sys#make-c-string str #!optional (loc '##sys#make-c-string)) + (let* ([len (##sys#size str)] + [buf (##sys#make-string (fx+ len 1))] ) + (##core#inline "C_substring_copy" str buf 0 len 0) + (##core#inline "C_setsubchar" buf len #\nul) + (if (fx= (##core#inline "C_asciiz_strlen" buf) len) + buf + (##sys#signal-hook #:type-error loc + "cannot represent string with NUL bytes as C string" + str))) ) (define ##sys#peek-signed-integer (##core#primitive "C_peek_signed_integer")) (define ##sys#peek-unsigned-integer (##core#primitive "C_peek_unsigned_integer")) diff --git a/tests/compiler-tests.scm b/tests/compiler-tests.scm index 65ebcb89..923c4358 100644 --- a/tests/compiler-tests.scm +++ b/tests/compiler-tests.scm @@ -123,3 +123,12 @@ (set! x #t) f1))))) (for-each f1 '(1 2 3)))) + +(newline) + +;; Test safety of ##sys#make-c-string +(handle-exceptions exn (print "Good, unrepresentable C strings cause errors") + (print "BUG! We got, without error, length = " + ((foreign-lambda* int ((c-string str)) + "C_return(strlen(str));") + "foo\x00bar")))Trap