aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/Makefile.dep35
-rw-r--r--src/lib_ffi.c3
-rw-r--r--src/lib_io.c3
-rw-r--r--src/lj_api.c5
-rw-r--r--src/lj_bcread.c3
-rw-r--r--src/lj_clib.c13
-rw-r--r--src/lj_cparse.c13
-rw-r--r--src/lj_debug.c15
-rw-r--r--src/lj_err.c21
-rw-r--r--src/lj_lex.c5
-rw-r--r--src/lj_str.c66
-rw-r--r--src/lj_str.h8
-rw-r--r--src/lj_strfmt.c76
-rw-r--r--src/lj_strfmt.h9
14 files changed, 149 insertions, 126 deletions
diff --git a/src/Makefile.dep b/src/Makefile.dep
index b2ced9c9..70109c82 100644
--- a/src/Makefile.dep
+++ b/src/Makefile.dep
@@ -16,11 +16,12 @@ lib_debug.o: lib_debug.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h \
16lib_ffi.o: lib_ffi.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \ 16lib_ffi.o: lib_ffi.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
17 lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_meta.h \ 17 lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_meta.h \
18 lj_ctype.h lj_cparse.h lj_cdata.h lj_cconv.h lj_carith.h lj_ccall.h \ 18 lj_ctype.h lj_cparse.h lj_cdata.h lj_cconv.h lj_carith.h lj_ccall.h \
19 lj_ccallback.h lj_clib.h lj_ff.h lj_ffdef.h lj_lib.h lj_libdef.h 19 lj_ccallback.h lj_clib.h lj_strfmt.h lj_ff.h lj_ffdef.h lj_lib.h \
20 lj_libdef.h
20lib_init.o: lib_init.c lua.h luaconf.h lauxlib.h lualib.h lj_arch.h 21lib_init.o: lib_init.c lua.h luaconf.h lauxlib.h lualib.h lj_arch.h
21lib_io.o: lib_io.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \ 22lib_io.o: lib_io.c lua.h luaconf.h lauxlib.h lualib.h lj_obj.h lj_def.h \
22 lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_state.h \ 23 lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_state.h \
23 lj_ff.h lj_ffdef.h lj_lib.h lj_libdef.h 24 lj_strfmt.h lj_ff.h lj_ffdef.h lj_lib.h lj_libdef.h
24lib_jit.o: lib_jit.c lua.h luaconf.h lauxlib.h lualib.h lj_arch.h \ 25lib_jit.o: lib_jit.c lua.h luaconf.h lauxlib.h lualib.h lj_arch.h \
25 lj_obj.h lj_def.h lj_err.h lj_errmsg.h lj_debug.h lj_str.h lj_tab.h \ 26 lj_obj.h lj_def.h lj_err.h lj_errmsg.h lj_debug.h lj_str.h lj_tab.h \
26 lj_bc.h lj_ir.h lj_jit.h lj_ircall.h lj_iropt.h lj_target.h \ 27 lj_bc.h lj_ir.h lj_jit.h lj_ircall.h lj_iropt.h lj_target.h \
@@ -44,7 +45,7 @@ lj_alloc.o: lj_alloc.c lj_def.h lua.h luaconf.h lj_arch.h lj_alloc.h
44lj_api.o: lj_api.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \ 45lj_api.o: lj_api.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
45 lj_err.h lj_errmsg.h lj_debug.h lj_str.h lj_tab.h lj_func.h lj_udata.h \ 46 lj_err.h lj_errmsg.h lj_debug.h lj_str.h lj_tab.h lj_func.h lj_udata.h \
46 lj_meta.h lj_state.h lj_bc.h lj_frame.h lj_trace.h lj_jit.h lj_ir.h \ 47 lj_meta.h lj_state.h lj_bc.h lj_frame.h lj_trace.h lj_jit.h lj_ir.h \
47 lj_dispatch.h lj_traceerr.h lj_vm.h lj_strscan.h 48 lj_dispatch.h lj_traceerr.h lj_vm.h lj_strscan.h lj_strfmt.h
48lj_asm.o: lj_asm.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \ 49lj_asm.o: lj_asm.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
49 lj_str.h lj_tab.h lj_frame.h lj_bc.h lj_ctype.h lj_ir.h lj_jit.h \ 50 lj_str.h lj_tab.h lj_frame.h lj_bc.h lj_ctype.h lj_ir.h lj_jit.h \
50 lj_ircall.h lj_iropt.h lj_mcode.h lj_trace.h lj_dispatch.h lj_traceerr.h \ 51 lj_ircall.h lj_iropt.h lj_mcode.h lj_trace.h lj_dispatch.h lj_traceerr.h \
@@ -54,7 +55,8 @@ lj_bc.o: lj_bc.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_bc.h \
54 lj_bcdef.h 55 lj_bcdef.h
55lj_bcread.o: lj_bcread.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 56lj_bcread.o: lj_bcread.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
56 lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_tab.h lj_bc.h \ 57 lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_tab.h lj_bc.h \
57 lj_ctype.h lj_cdata.h lualib.h lj_lex.h lj_bcdump.h lj_state.h 58 lj_ctype.h lj_cdata.h lualib.h lj_lex.h lj_bcdump.h lj_state.h \
59 lj_strfmt.h
58lj_bcwrite.o: lj_bcwrite.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 60lj_bcwrite.o: lj_bcwrite.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
59 lj_gc.h lj_buf.h lj_str.h lj_bc.h lj_ctype.h lj_dispatch.h lj_jit.h \ 61 lj_gc.h lj_buf.h lj_str.h lj_bc.h lj_ctype.h lj_dispatch.h lj_jit.h \
60 lj_ir.h lj_bcdump.h lj_lex.h lj_err.h lj_errmsg.h lj_vm.h 62 lj_ir.h lj_bcdump.h lj_lex.h lj_err.h lj_errmsg.h lj_vm.h
@@ -81,10 +83,10 @@ lj_cdata.o: lj_cdata.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
81lj_char.o: lj_char.c lj_char.h lj_def.h lua.h luaconf.h 83lj_char.o: lj_char.c lj_char.h lj_def.h lua.h luaconf.h
82lj_clib.o: lj_clib.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \ 84lj_clib.o: lj_clib.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
83 lj_err.h lj_errmsg.h lj_tab.h lj_str.h lj_udata.h lj_ctype.h lj_cconv.h \ 85 lj_err.h lj_errmsg.h lj_tab.h lj_str.h lj_udata.h lj_ctype.h lj_cconv.h \
84 lj_cdata.h lj_clib.h 86 lj_cdata.h lj_clib.h lj_strfmt.h
85lj_cparse.o: lj_cparse.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 87lj_cparse.o: lj_cparse.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
86 lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_ctype.h lj_cparse.h \ 88 lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_ctype.h lj_cparse.h \
87 lj_frame.h lj_bc.h lj_vm.h lj_char.h lj_strscan.h 89 lj_frame.h lj_bc.h lj_vm.h lj_char.h lj_strscan.h lj_strfmt.h
88lj_crecord.o: lj_crecord.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 90lj_crecord.o: lj_crecord.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
89 lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_frame.h lj_bc.h lj_ctype.h \ 91 lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_frame.h lj_bc.h lj_ctype.h \
90 lj_gc.h lj_cdata.h lj_cparse.h lj_cconv.h lj_clib.h lj_ccall.h lj_ff.h \ 92 lj_gc.h lj_cdata.h lj_cparse.h lj_cconv.h lj_clib.h lj_ccall.h lj_ff.h \
@@ -95,7 +97,7 @@ lj_ctype.o: lj_ctype.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
95 lj_gc.h lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_ctype.h lj_ccallback.h 97 lj_gc.h lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_ctype.h lj_ccallback.h
96lj_debug.o: lj_debug.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 98lj_debug.o: lj_debug.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
97 lj_err.h lj_errmsg.h lj_debug.h lj_buf.h lj_gc.h lj_str.h lj_tab.h \ 99 lj_err.h lj_errmsg.h lj_debug.h lj_buf.h lj_gc.h lj_str.h lj_tab.h \
98 lj_state.h lj_frame.h lj_bc.h lj_jit.h lj_ir.h 100 lj_state.h lj_frame.h lj_bc.h lj_strfmt.h lj_jit.h lj_ir.h
99lj_dispatch.o: lj_dispatch.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 101lj_dispatch.o: lj_dispatch.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
100 lj_err.h lj_errmsg.h lj_buf.h lj_gc.h lj_str.h lj_func.h lj_tab.h \ 102 lj_err.h lj_errmsg.h lj_buf.h lj_gc.h lj_str.h lj_func.h lj_tab.h \
101 lj_meta.h lj_debug.h lj_state.h lj_frame.h lj_bc.h lj_ff.h lj_ffdef.h \ 103 lj_meta.h lj_debug.h lj_state.h lj_frame.h lj_bc.h lj_ff.h lj_ffdef.h \
@@ -104,7 +106,7 @@ lj_dispatch.o: lj_dispatch.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
104lj_err.o: lj_err.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_err.h \ 106lj_err.o: lj_err.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_err.h \
105 lj_errmsg.h lj_debug.h lj_str.h lj_func.h lj_state.h lj_frame.h lj_bc.h \ 107 lj_errmsg.h lj_debug.h lj_str.h lj_func.h lj_state.h lj_frame.h lj_bc.h \
106 lj_ff.h lj_ffdef.h lj_trace.h lj_jit.h lj_ir.h lj_dispatch.h \ 108 lj_ff.h lj_ffdef.h lj_trace.h lj_jit.h lj_ir.h lj_dispatch.h \
107 lj_traceerr.h lj_vm.h 109 lj_traceerr.h lj_vm.h lj_strfmt.h
108lj_ffrecord.o: lj_ffrecord.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 110lj_ffrecord.o: lj_ffrecord.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
109 lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_frame.h lj_bc.h lj_ff.h \ 111 lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_frame.h lj_bc.h lj_ff.h \
110 lj_ffdef.h lj_ir.h lj_jit.h lj_ircall.h lj_iropt.h lj_trace.h \ 112 lj_ffdef.h lj_ir.h lj_jit.h lj_ircall.h lj_iropt.h lj_trace.h \
@@ -126,7 +128,8 @@ lj_ir.o: lj_ir.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
126 lj_carith.h lj_vm.h lj_strscan.h lj_lib.h 128 lj_carith.h lj_vm.h lj_strscan.h lj_lib.h
127lj_lex.o: lj_lex.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \ 129lj_lex.o: lj_lex.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
128 lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_tab.h lj_ctype.h lj_cdata.h \ 130 lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_tab.h lj_ctype.h lj_cdata.h \
129 lualib.h lj_state.h lj_lex.h lj_parse.h lj_char.h lj_strscan.h 131 lualib.h lj_state.h lj_lex.h lj_parse.h lj_char.h lj_strscan.h \
132 lj_strfmt.h
130lj_lib.o: lj_lib.c lauxlib.h lua.h luaconf.h lj_obj.h lj_def.h lj_arch.h \ 133lj_lib.o: lj_lib.c lauxlib.h lua.h luaconf.h lj_obj.h lj_def.h lj_arch.h \
131 lj_gc.h lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_func.h lj_bc.h \ 134 lj_gc.h lj_err.h lj_errmsg.h lj_str.h lj_tab.h lj_func.h lj_bc.h \
132 lj_dispatch.h lj_jit.h lj_ir.h lj_vm.h lj_strscan.h lj_lex.h lj_bcdump.h \ 135 lj_dispatch.h lj_jit.h lj_ir.h lj_vm.h lj_strscan.h lj_lex.h lj_bcdump.h \
@@ -179,9 +182,9 @@ lj_state.o: lj_state.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
179 lj_meta.h lj_state.h lj_frame.h lj_bc.h lj_ctype.h lj_trace.h lj_jit.h \ 182 lj_meta.h lj_state.h lj_frame.h lj_bc.h lj_ctype.h lj_trace.h lj_jit.h \
180 lj_ir.h lj_dispatch.h lj_traceerr.h lj_vm.h lj_lex.h lj_alloc.h 183 lj_ir.h lj_dispatch.h lj_traceerr.h lj_vm.h lj_lex.h lj_alloc.h
181lj_str.o: lj_str.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \ 184lj_str.o: lj_str.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
182 lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_state.h lj_char.h 185 lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_char.h
183lj_strfmt.o: lj_strfmt.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 186lj_strfmt.o: lj_strfmt.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
184 lj_buf.h lj_gc.h lj_str.h lj_char.h lj_strfmt.h 187 lj_buf.h lj_gc.h lj_str.h lj_state.h lj_char.h lj_strfmt.h
185lj_strscan.o: lj_strscan.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \ 188lj_strscan.o: lj_strscan.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h \
186 lj_char.h lj_strscan.h 189 lj_char.h lj_strscan.h
187lj_tab.o: lj_tab.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \ 190lj_tab.o: lj_tab.c lj_obj.h lua.h luaconf.h lj_def.h lj_arch.h lj_gc.h \
@@ -202,11 +205,11 @@ ljamalg.o: ljamalg.c lua.h luaconf.h lauxlib.h lj_gc.c lj_obj.h lj_def.h \
202 lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_tab.h \ 205 lj_arch.h lj_gc.h lj_err.h lj_errmsg.h lj_buf.h lj_str.h lj_tab.h \
203 lj_func.h lj_udata.h lj_meta.h lj_state.h lj_frame.h lj_bc.h lj_ctype.h \ 206 lj_func.h lj_udata.h lj_meta.h lj_state.h lj_frame.h lj_bc.h lj_ctype.h \
204 lj_cdata.h lj_trace.h lj_jit.h lj_ir.h lj_dispatch.h lj_traceerr.h \ 207 lj_cdata.h lj_trace.h lj_jit.h lj_ir.h lj_dispatch.h lj_traceerr.h \
205 lj_vm.h lj_err.c lj_debug.h lj_ff.h lj_ffdef.h lj_char.c lj_char.h \ 208 lj_vm.h lj_err.c lj_debug.h lj_ff.h lj_ffdef.h lj_strfmt.h lj_char.c \
206 lj_bc.c lj_bcdef.h lj_obj.c lj_buf.c lj_str.c lj_tab.c lj_func.c \ 209 lj_char.h lj_bc.c lj_bcdef.h lj_obj.c lj_buf.c lj_str.c lj_tab.c \
207 lj_udata.c lj_meta.c lj_strscan.h lj_lib.h lj_debug.c lj_state.c \ 210 lj_func.c lj_udata.c lj_meta.c lj_strscan.h lj_lib.h lj_debug.c \
208 lj_lex.h lj_alloc.h lj_dispatch.c lj_ccallback.h luajit.h lj_vmevent.c \ 211 lj_state.c lj_lex.h lj_alloc.h lj_dispatch.c lj_ccallback.h luajit.h \
209 lj_vmevent.h lj_vmmath.c lj_strscan.c lj_strfmt.c lj_strfmt.h lj_api.c \ 212 lj_vmevent.c lj_vmevent.h lj_vmmath.c lj_strscan.c lj_strfmt.c lj_api.c \
210 lj_lex.c lualib.h lj_parse.h lj_parse.c lj_bcread.c lj_bcdump.h \ 213 lj_lex.c lualib.h lj_parse.h lj_parse.c lj_bcread.c lj_bcdump.h \
211 lj_bcwrite.c lj_load.c lj_ctype.c lj_cdata.c lj_cconv.h lj_cconv.c \ 214 lj_bcwrite.c lj_load.c lj_ctype.c lj_cdata.c lj_cconv.h lj_cconv.c \
212 lj_ccall.c lj_ccall.h lj_ccallback.c lj_target.h lj_target_*.h \ 215 lj_ccall.c lj_ccall.h lj_ccallback.c lj_target.h lj_target_*.h \
diff --git a/src/lib_ffi.c b/src/lib_ffi.c
index 46f27e01..2ac8290b 100644
--- a/src/lib_ffi.c
+++ b/src/lib_ffi.c
@@ -29,6 +29,7 @@
29#include "lj_ccall.h" 29#include "lj_ccall.h"
30#include "lj_ccallback.h" 30#include "lj_ccallback.h"
31#include "lj_clib.h" 31#include "lj_clib.h"
32#include "lj_strfmt.h"
32#include "lj_ff.h" 33#include "lj_ff.h"
33#include "lj_lib.h" 34#include "lj_lib.h"
34 35
@@ -317,7 +318,7 @@ LJLIB_CF(ffi_meta___tostring)
317 } 318 }
318 } 319 }
319 } 320 }
320 lj_str_pushf(L, msg, strdata(lj_ctype_repr(L, id, NULL)), p); 321 lj_strfmt_pushf(L, msg, strdata(lj_ctype_repr(L, id, NULL)), p);
321checkgc: 322checkgc:
322 lj_gc_check(L); 323 lj_gc_check(L);
323 return 1; 324 return 1;
diff --git a/src/lib_io.c b/src/lib_io.c
index 18d87a89..cae56579 100644
--- a/src/lib_io.c
+++ b/src/lib_io.c
@@ -22,6 +22,7 @@
22#include "lj_buf.h" 22#include "lj_buf.h"
23#include "lj_str.h" 23#include "lj_str.h"
24#include "lj_state.h" 24#include "lj_state.h"
25#include "lj_strfmt.h"
25#include "lj_ff.h" 26#include "lj_ff.h"
26#include "lj_lib.h" 27#include "lj_lib.h"
27 28
@@ -85,7 +86,7 @@ static IOFileUD *io_file_open(lua_State *L, const char *mode)
85 IOFileUD *iof = io_file_new(L); 86 IOFileUD *iof = io_file_new(L);
86 iof->fp = fopen(fname, mode); 87 iof->fp = fopen(fname, mode);
87 if (iof->fp == NULL) 88 if (iof->fp == NULL)
88 luaL_argerror(L, 1, lj_str_pushf(L, "%s: %s", fname, strerror(errno))); 89 luaL_argerror(L, 1, lj_strfmt_pushf(L, "%s: %s", fname, strerror(errno)));
89 return iof; 90 return iof;
90} 91}
91 92
diff --git a/src/lj_api.c b/src/lj_api.c
index edb2d620..451e4444 100644
--- a/src/lj_api.c
+++ b/src/lj_api.c
@@ -24,6 +24,7 @@
24#include "lj_trace.h" 24#include "lj_trace.h"
25#include "lj_vm.h" 25#include "lj_vm.h"
26#include "lj_strscan.h" 26#include "lj_strscan.h"
27#include "lj_strfmt.h"
27 28
28/* -- Common helper functions --------------------------------------------- */ 29/* -- Common helper functions --------------------------------------------- */
29 30
@@ -606,7 +607,7 @@ LUA_API const char *lua_pushvfstring(lua_State *L, const char *fmt,
606 va_list argp) 607 va_list argp)
607{ 608{
608 lj_gc_check(L); 609 lj_gc_check(L);
609 return lj_str_pushvf(L, fmt, argp); 610 return lj_strfmt_pushvf(L, fmt, argp);
610} 611}
611 612
612LUA_API const char *lua_pushfstring(lua_State *L, const char *fmt, ...) 613LUA_API const char *lua_pushfstring(lua_State *L, const char *fmt, ...)
@@ -615,7 +616,7 @@ LUA_API const char *lua_pushfstring(lua_State *L, const char *fmt, ...)
615 va_list argp; 616 va_list argp;
616 lj_gc_check(L); 617 lj_gc_check(L);
617 va_start(argp, fmt); 618 va_start(argp, fmt);
618 ret = lj_str_pushvf(L, fmt, argp); 619 ret = lj_strfmt_pushvf(L, fmt, argp);
619 va_end(argp); 620 va_end(argp);
620 return ret; 621 return ret;
621} 622}
diff --git a/src/lj_bcread.c b/src/lj_bcread.c
index 7bb16a60..9f025500 100644
--- a/src/lj_bcread.c
+++ b/src/lj_bcread.c
@@ -21,6 +21,7 @@
21#include "lj_lex.h" 21#include "lj_lex.h"
22#include "lj_bcdump.h" 22#include "lj_bcdump.h"
23#include "lj_state.h" 23#include "lj_state.h"
24#include "lj_strfmt.h"
24 25
25/* Reuse some lexer fields for our own purposes. */ 26/* Reuse some lexer fields for our own purposes. */
26#define bcread_flags(ls) ls->level 27#define bcread_flags(ls) ls->level
@@ -39,7 +40,7 @@ static LJ_NOINLINE void bcread_error(LexState *ls, ErrMsg em)
39 const char *name = ls->chunkarg; 40 const char *name = ls->chunkarg;
40 if (*name == BCDUMP_HEAD1) name = "(binary)"; 41 if (*name == BCDUMP_HEAD1) name = "(binary)";
41 else if (*name == '@' || *name == '=') name++; 42 else if (*name == '@' || *name == '=') name++;
42 lj_str_pushf(L, "%s: %s", name, err2msg(em)); 43 lj_strfmt_pushf(L, "%s: %s", name, err2msg(em));
43 lj_err_throw(L, LUA_ERRSYNTAX); 44 lj_err_throw(L, LUA_ERRSYNTAX);
44} 45}
45 46
diff --git a/src/lj_clib.c b/src/lj_clib.c
index 23d1f182..263028e4 100644
--- a/src/lj_clib.c
+++ b/src/lj_clib.c
@@ -16,6 +16,7 @@
16#include "lj_cconv.h" 16#include "lj_cconv.h"
17#include "lj_cdata.h" 17#include "lj_cdata.h"
18#include "lj_clib.h" 18#include "lj_clib.h"
19#include "lj_strfmt.h"
19 20
20/* -- OS-specific functions ----------------------------------------------- */ 21/* -- OS-specific functions ----------------------------------------------- */
21 22
@@ -61,7 +62,7 @@ static const char *clib_extname(lua_State *L, const char *name)
61#endif 62#endif
62 ) { 63 ) {
63 if (!strchr(name, '.')) { 64 if (!strchr(name, '.')) {
64 name = lj_str_pushf(L, CLIB_SOEXT, name); 65 name = lj_strfmt_pushf(L, CLIB_SOEXT, name);
65 L->top--; 66 L->top--;
66#ifdef __CYGWIN__ 67#ifdef __CYGWIN__
67 } else { 68 } else {
@@ -70,7 +71,7 @@ static const char *clib_extname(lua_State *L, const char *name)
70 } 71 }
71 if (!(name[0] == CLIB_SOPREFIX[0] && name[1] == CLIB_SOPREFIX[1] && 72 if (!(name[0] == CLIB_SOPREFIX[0] && name[1] == CLIB_SOPREFIX[1] &&
72 name[2] == CLIB_SOPREFIX[2])) { 73 name[2] == CLIB_SOPREFIX[2])) {
73 name = lj_str_pushf(L, CLIB_SOPREFIX "%s", name); 74 name = lj_strfmt_pushf(L, CLIB_SOPREFIX "%s", name);
74 L->top--; 75 L->top--;
75 } 76 }
76 } 77 }
@@ -178,7 +179,7 @@ LJ_NORET LJ_NOINLINE static void clib_error(lua_State *L, const char *fmt,
178 if (!FormatMessageA(FORMAT_MESSAGE_IGNORE_INSERTS|FORMAT_MESSAGE_FROM_SYSTEM, 179 if (!FormatMessageA(FORMAT_MESSAGE_IGNORE_INSERTS|FORMAT_MESSAGE_FROM_SYSTEM,
179 NULL, err, 0, buf, sizeof(buf), NULL)) 180 NULL, err, 0, buf, sizeof(buf), NULL))
180 buf[0] = '\0'; 181 buf[0] = '\0';
181 lj_err_callermsg(L, lj_str_pushf(L, fmt, name, buf)); 182 lj_err_callermsg(L, lj_strfmt_pushf(L, fmt, name, buf));
182} 183}
183 184
184static int clib_needext(const char *s) 185static int clib_needext(const char *s)
@@ -193,7 +194,7 @@ static int clib_needext(const char *s)
193static const char *clib_extname(lua_State *L, const char *name) 194static const char *clib_extname(lua_State *L, const char *name)
194{ 195{
195 if (clib_needext(name)) { 196 if (clib_needext(name)) {
196 name = lj_str_pushf(L, "%s.dll", name); 197 name = lj_strfmt_pushf(L, "%s.dll", name);
197 L->top--; 198 L->top--;
198 } 199 }
199 return name; 200 return name;
@@ -266,7 +267,7 @@ static void *clib_getsym(CLibrary *cl, const char *name)
266LJ_NORET LJ_NOINLINE static void clib_error(lua_State *L, const char *fmt, 267LJ_NORET LJ_NOINLINE static void clib_error(lua_State *L, const char *fmt,
267 const char *name) 268 const char *name)
268{ 269{
269 lj_err_callermsg(L, lj_str_pushf(L, fmt, name, "no support for this OS")); 270 lj_err_callermsg(L, lj_strfmt_pushf(L, fmt, name, "no support for this OS"));
270} 271}
271 272
272static void *clib_loadlib(lua_State *L, const char *name, int global) 273static void *clib_loadlib(lua_State *L, const char *name, int global)
@@ -350,7 +351,7 @@ TValue *lj_clib_index(lua_State *L, CLibrary *cl, GCstr *name)
350 CTInfo cconv = ctype_cconv(ct->info); 351 CTInfo cconv = ctype_cconv(ct->info);
351 if (cconv == CTCC_FASTCALL || cconv == CTCC_STDCALL) { 352 if (cconv == CTCC_FASTCALL || cconv == CTCC_STDCALL) {
352 CTSize sz = clib_func_argsize(cts, ct); 353 CTSize sz = clib_func_argsize(cts, ct);
353 const char *symd = lj_str_pushf(L, 354 const char *symd = lj_strfmt_pushf(L,
354 cconv == CTCC_FASTCALL ? "@%s@%d" : "_%s@%d", 355 cconv == CTCC_FASTCALL ? "@%s@%d" : "_%s@%d",
355 sym, sz); 356 sym, sz);
356 L->top--; 357 L->top--;
diff --git a/src/lj_cparse.c b/src/lj_cparse.c
index b8c95bd3..484a2c15 100644
--- a/src/lj_cparse.c
+++ b/src/lj_cparse.c
@@ -17,6 +17,7 @@
17#include "lj_vm.h" 17#include "lj_vm.h"
18#include "lj_char.h" 18#include "lj_char.h"
19#include "lj_strscan.h" 19#include "lj_strscan.h"
20#include "lj_strfmt.h"
20 21
21/* 22/*
22** Important note: this is NOT a validating C parser! This is a minimal 23** Important note: this is NOT a validating C parser! This is a minimal
@@ -47,9 +48,9 @@ static const char *cp_tok2str(CPState *cp, CPToken tok)
47 if (tok > CTOK_OFS) 48 if (tok > CTOK_OFS)
48 return ctoknames[tok-CTOK_OFS-1]; 49 return ctoknames[tok-CTOK_OFS-1];
49 else if (!lj_char_iscntrl(tok)) 50 else if (!lj_char_iscntrl(tok))
50 return lj_str_pushf(cp->L, "%c", tok); 51 return lj_strfmt_pushf(cp->L, "%c", tok);
51 else 52 else
52 return lj_str_pushf(cp->L, "char(%d)", tok); 53 return lj_strfmt_pushf(cp->L, "char(%d)", tok);
53} 54}
54 55
55/* End-of-line? */ 56/* End-of-line? */
@@ -117,12 +118,12 @@ LJ_NORET static void cp_errmsg(CPState *cp, CPToken tok, ErrMsg em, ...)
117 } 118 }
118 L = cp->L; 119 L = cp->L;
119 va_start(argp, em); 120 va_start(argp, em);
120 msg = lj_str_pushvf(L, err2msg(em), argp); 121 msg = lj_strfmt_pushvf(L, err2msg(em), argp);
121 va_end(argp); 122 va_end(argp);
122 if (tokstr) 123 if (tokstr)
123 msg = lj_str_pushf(L, err2msg(LJ_ERR_XNEAR), msg, tokstr); 124 msg = lj_strfmt_pushf(L, err2msg(LJ_ERR_XNEAR), msg, tokstr);
124 if (cp->linenumber > 1) 125 if (cp->linenumber > 1)
125 msg = lj_str_pushf(L, "%s at line %d", msg, cp->linenumber); 126 msg = lj_strfmt_pushf(L, "%s at line %d", msg, cp->linenumber);
126 lj_err_callermsg(L, msg); 127 lj_err_callermsg(L, msg);
127} 128}
128 129
@@ -998,7 +999,7 @@ static void cp_decl_asm(CPState *cp, CPDecl *decl)
998 if (cp->tok == CTOK_STRING) { 999 if (cp->tok == CTOK_STRING) {
999 GCstr *str = cp->str; 1000 GCstr *str = cp->str;
1000 while (cp_next(cp) == CTOK_STRING) { 1001 while (cp_next(cp) == CTOK_STRING) {
1001 lj_str_pushf(cp->L, "%s%s", strdata(str), strdata(cp->str)); 1002 lj_strfmt_pushf(cp->L, "%s%s", strdata(str), strdata(cp->str));
1002 cp->L->top--; 1003 cp->L->top--;
1003 str = strV(cp->L->top); 1004 str = strV(cp->L->top);
1004 } 1005 }
diff --git a/src/lj_debug.c b/src/lj_debug.c
index 3f502864..48364069 100644
--- a/src/lj_debug.c
+++ b/src/lj_debug.c
@@ -15,6 +15,7 @@
15#include "lj_state.h" 15#include "lj_state.h"
16#include "lj_frame.h" 16#include "lj_frame.h"
17#include "lj_bc.h" 17#include "lj_bc.h"
18#include "lj_strfmt.h"
18#if LJ_HASJIT 19#if LJ_HASJIT
19#include "lj_jit.h" 20#include "lj_jit.h"
20#endif 21#endif
@@ -350,12 +351,12 @@ void lj_debug_addloc(lua_State *L, const char *msg,
350 GCproto *pt = funcproto(fn); 351 GCproto *pt = funcproto(fn);
351 char buf[LUA_IDSIZE]; 352 char buf[LUA_IDSIZE];
352 lj_debug_shortname(buf, proto_chunkname(pt), pt->firstline); 353 lj_debug_shortname(buf, proto_chunkname(pt), pt->firstline);
353 lj_str_pushf(L, "%s:%d: %s", buf, line, msg); 354 lj_strfmt_pushf(L, "%s:%d: %s", buf, line, msg);
354 return; 355 return;
355 } 356 }
356 } 357 }
357 } 358 }
358 lj_str_pushf(L, "%s", msg); 359 lj_strfmt_pushf(L, "%s", msg);
359} 360}
360 361
361/* Push location string for a bytecode position to Lua stack. */ 362/* Push location string for a bytecode position to Lua stack. */
@@ -366,7 +367,7 @@ void lj_debug_pushloc(lua_State *L, GCproto *pt, BCPos pc)
366 MSize i, len = name->len; 367 MSize i, len = name->len;
367 BCLine line = lj_debug_line(pt, pc); 368 BCLine line = lj_debug_line(pt, pc);
368 if (pt->firstline == ~(BCLine)0) { 369 if (pt->firstline == ~(BCLine)0) {
369 lj_str_pushf(L, "builtin:%s", s); 370 lj_strfmt_pushf(L, "builtin:%s", s);
370 } else if (*s == '@') { 371 } else if (*s == '@') {
371 s++; len--; 372 s++; len--;
372 for (i = len; i > 0; i--) 373 for (i = len; i > 0; i--)
@@ -374,13 +375,13 @@ void lj_debug_pushloc(lua_State *L, GCproto *pt, BCPos pc)
374 s += i+1; 375 s += i+1;
375 break; 376 break;
376 } 377 }
377 lj_str_pushf(L, "%s:%d", s, line); 378 lj_strfmt_pushf(L, "%s:%d", s, line);
378 } else if (len > 40) { 379 } else if (len > 40) {
379 lj_str_pushf(L, "%p:%d", pt, line); 380 lj_strfmt_pushf(L, "%p:%d", pt, line);
380 } else if (*s == '=') { 381 } else if (*s == '=') {
381 lj_str_pushf(L, "%s:%d", s+1, line); 382 lj_strfmt_pushf(L, "%s:%d", s+1, line);
382 } else { 383 } else {
383 lj_str_pushf(L, "\"%s\":%d", s, line); 384 lj_strfmt_pushf(L, "\"%s\":%d", s, line);
384 } 385 }
385} 386}
386 387
diff --git a/src/lj_err.c b/src/lj_err.c
index e0fb7167..8ff0a455 100644
--- a/src/lj_err.c
+++ b/src/lj_err.c
@@ -16,6 +16,7 @@
16#include "lj_ff.h" 16#include "lj_ff.h"
17#include "lj_trace.h" 17#include "lj_trace.h"
18#include "lj_vm.h" 18#include "lj_vm.h"
19#include "lj_strfmt.h"
19 20
20/* 21/*
21** LuaJIT can either use internal or external frame unwinding: 22** LuaJIT can either use internal or external frame unwinding:
@@ -569,7 +570,7 @@ LJ_NORET LJ_NOINLINE static void err_msgv(lua_State *L, ErrMsg em, ...)
569 va_list argp; 570 va_list argp;
570 va_start(argp, em); 571 va_start(argp, em);
571 if (curr_funcisL(L)) L->top = curr_topL(L); 572 if (curr_funcisL(L)) L->top = curr_topL(L);
572 msg = lj_str_pushvf(L, err2msg(em), argp); 573 msg = lj_strfmt_pushvf(L, err2msg(em), argp);
573 va_end(argp); 574 va_end(argp);
574 lj_debug_addloc(L, msg, L->base-1, NULL); 575 lj_debug_addloc(L, msg, L->base-1, NULL);
575 lj_err_run(L); 576 lj_err_run(L);
@@ -588,10 +589,10 @@ LJ_NOINLINE void lj_err_lex(lua_State *L, GCstr *src, const char *tok,
588 char buff[LUA_IDSIZE]; 589 char buff[LUA_IDSIZE];
589 const char *msg; 590 const char *msg;
590 lj_debug_shortname(buff, src, line); 591 lj_debug_shortname(buff, src, line);
591 msg = lj_str_pushvf(L, err2msg(em), argp); 592 msg = lj_strfmt_pushvf(L, err2msg(em), argp);
592 msg = lj_str_pushf(L, "%s:%d: %s", buff, line, msg); 593 msg = lj_strfmt_pushf(L, "%s:%d: %s", buff, line, msg);
593 if (tok) 594 if (tok)
594 lj_str_pushf(L, err2msg(LJ_ERR_XNEAR), msg, tok); 595 lj_strfmt_pushf(L, err2msg(LJ_ERR_XNEAR), msg, tok);
595 lj_err_throw(L, LUA_ERRSYNTAX); 596 lj_err_throw(L, LUA_ERRSYNTAX);
596} 597}
597 598
@@ -675,7 +676,7 @@ LJ_NOINLINE void lj_err_callerv(lua_State *L, ErrMsg em, ...)
675 const char *msg; 676 const char *msg;
676 va_list argp; 677 va_list argp;
677 va_start(argp, em); 678 va_start(argp, em);
678 msg = lj_str_pushvf(L, err2msg(em), argp); 679 msg = lj_strfmt_pushvf(L, err2msg(em), argp);
679 va_end(argp); 680 va_end(argp);
680 lj_err_callermsg(L, msg); 681 lj_err_callermsg(L, msg);
681} 682}
@@ -695,9 +696,9 @@ LJ_NORET LJ_NOINLINE static void err_argmsg(lua_State *L, int narg,
695 if (narg < 0 && narg > LUA_REGISTRYINDEX) 696 if (narg < 0 && narg > LUA_REGISTRYINDEX)
696 narg = (int)(L->top - L->base) + narg + 1; 697 narg = (int)(L->top - L->base) + narg + 1;
697 if (ftype && ftype[3] == 'h' && --narg == 0) /* Check for "method". */ 698 if (ftype && ftype[3] == 'h' && --narg == 0) /* Check for "method". */
698 msg = lj_str_pushf(L, err2msg(LJ_ERR_BADSELF), fname, msg); 699 msg = lj_strfmt_pushf(L, err2msg(LJ_ERR_BADSELF), fname, msg);
699 else 700 else
700 msg = lj_str_pushf(L, err2msg(LJ_ERR_BADARG), narg, fname, msg); 701 msg = lj_strfmt_pushf(L, err2msg(LJ_ERR_BADARG), narg, fname, msg);
701 lj_err_callermsg(L, msg); 702 lj_err_callermsg(L, msg);
702} 703}
703 704
@@ -707,7 +708,7 @@ LJ_NOINLINE void lj_err_argv(lua_State *L, int narg, ErrMsg em, ...)
707 const char *msg; 708 const char *msg;
708 va_list argp; 709 va_list argp;
709 va_start(argp, em); 710 va_start(argp, em);
710 msg = lj_str_pushvf(L, err2msg(em), argp); 711 msg = lj_strfmt_pushvf(L, err2msg(em), argp);
711 va_end(argp); 712 va_end(argp);
712 err_argmsg(L, narg, msg); 713 err_argmsg(L, narg, msg);
713} 714}
@@ -723,7 +724,7 @@ LJ_NOINLINE void lj_err_argtype(lua_State *L, int narg, const char *xname)
723{ 724{
724 TValue *o = narg < 0 ? L->top + narg : L->base + narg-1; 725 TValue *o = narg < 0 ? L->top + narg : L->base + narg-1;
725 const char *tname = o < L->top ? lj_typename(o) : lj_obj_typename[0]; 726 const char *tname = o < L->top ? lj_typename(o) : lj_obj_typename[0];
726 const char *msg = lj_str_pushf(L, err2msg(LJ_ERR_BADTYPE), xname, tname); 727 const char *msg = lj_strfmt_pushf(L, err2msg(LJ_ERR_BADTYPE), xname, tname);
727 err_argmsg(L, narg, msg); 728 err_argmsg(L, narg, msg);
728} 729}
729 730
@@ -773,7 +774,7 @@ LUALIB_API int luaL_error(lua_State *L, const char *fmt, ...)
773 const char *msg; 774 const char *msg;
774 va_list argp; 775 va_list argp;
775 va_start(argp, fmt); 776 va_start(argp, fmt);
776 msg = lj_str_pushvf(L, fmt, argp); 777 msg = lj_strfmt_pushvf(L, fmt, argp);
777 va_end(argp); 778 va_end(argp);
778 lj_err_callermsg(L, msg); 779 lj_err_callermsg(L, msg);
779 return 0; /* unreachable */ 780 return 0; /* unreachable */
diff --git a/src/lj_lex.c b/src/lj_lex.c
index c988a6c1..7c2c6677 100644
--- a/src/lj_lex.c
+++ b/src/lj_lex.c
@@ -25,6 +25,7 @@
25#include "lj_parse.h" 25#include "lj_parse.h"
26#include "lj_char.h" 26#include "lj_char.h"
27#include "lj_strscan.h" 27#include "lj_strscan.h"
28#include "lj_strfmt.h"
28 29
29/* Lua lexer token names. */ 30/* Lua lexer token names. */
30static const char *const tokennames[] = { 31static const char *const tokennames[] = {
@@ -444,9 +445,9 @@ const char *lj_lex_token2str(LexState *ls, LexToken tok)
444 if (tok > TK_OFS) 445 if (tok > TK_OFS)
445 return tokennames[tok-TK_OFS-1]; 446 return tokennames[tok-TK_OFS-1];
446 else if (!lj_char_iscntrl(tok)) 447 else if (!lj_char_iscntrl(tok))
447 return lj_str_pushf(ls->L, "%c", tok); 448 return lj_strfmt_pushf(ls->L, "%c", tok);
448 else 449 else
449 return lj_str_pushf(ls->L, "char(%d)", tok); 450 return lj_strfmt_pushf(ls->L, "char(%d)", tok);
450} 451}
451 452
452/* Lexer error. */ 453/* Lexer error. */
diff --git a/src/lj_str.c b/src/lj_str.c
index f5bbae26..d21cecd0 100644
--- a/src/lj_str.c
+++ b/src/lj_str.c
@@ -13,7 +13,6 @@
13#include "lj_err.h" 13#include "lj_err.h"
14#include "lj_buf.h" 14#include "lj_buf.h"
15#include "lj_str.h" 15#include "lj_str.h"
16#include "lj_state.h"
17#include "lj_char.h" 16#include "lj_char.h"
18 17
19/* -- String helpers ------------------------------------------------------ */ 18/* -- String helpers ------------------------------------------------------ */
@@ -316,68 +315,3 @@ GCstr * LJ_FASTCALL lj_str_fromchar(lua_State *L, int c)
316 return lj_str_new(L, buf, 1); 315 return lj_str_new(L, buf, 1);
317} 316}
318 317
319/* -- String formatting --------------------------------------------------- */
320
321/* Push formatted message as a string object to Lua stack. va_list variant. */
322const char *lj_str_pushvf(lua_State *L, const char *fmt, va_list argp)
323{
324 SBuf *sb = &G(L)->tmpbuf;
325 setsbufL(sb, L);
326 lj_buf_need(sb, (MSize)strlen(fmt));
327 lj_buf_reset(sb);
328 for (;;) {
329 const char *e = strchr(fmt, '%');
330 if (e == NULL) break;
331 lj_buf_putmem(sb, fmt, (MSize)(e-fmt));
332 /* This function only handles %s, %c, %d, %f and %p formats. */
333 switch (e[1]) {
334 case 's': {
335 const char *s = va_arg(argp, char *);
336 if (s == NULL) s = "(null)";
337 lj_buf_putmem(sb, s, (MSize)strlen(s));
338 break;
339 }
340 case 'c':
341 lj_buf_putb(sb, va_arg(argp, int));
342 break;
343 case 'd':
344 setsbufP(sb, lj_str_bufint(lj_buf_more(sb, LJ_STR_INTBUF),
345 va_arg(argp, int32_t)));
346 break;
347 case 'f': {
348 TValue tv;
349 tv.n = va_arg(argp, lua_Number);
350 setsbufP(sb, lj_str_bufnum(lj_buf_more(sb, LJ_STR_NUMBUF), &tv));
351 break;
352 }
353 case 'p':
354 setsbufP(sb, lj_str_bufptr(lj_buf_more(sb, LJ_STR_PTRBUF),
355 va_arg(argp, void *)));
356 break;
357 case '%':
358 lj_buf_putb(sb, '%');
359 break;
360 default:
361 lj_buf_putb(sb, '%');
362 lj_buf_putb(sb, e[1]);
363 break;
364 }
365 fmt = e+2;
366 }
367 lj_buf_putmem(sb, fmt, (MSize)strlen(fmt));
368 setstrV(L, L->top, lj_buf_str(L, sb));
369 incr_top(L);
370 return strVdata(L->top - 1);
371}
372
373/* Push formatted message as a string object to Lua stack. Vararg variant. */
374const char *lj_str_pushf(lua_State *L, const char *fmt, ...)
375{
376 const char *msg;
377 va_list argp;
378 va_start(argp, fmt);
379 msg = lj_str_pushvf(L, fmt, argp);
380 va_end(argp);
381 return msg;
382}
383
diff --git a/src/lj_str.h b/src/lj_str.h
index 6e08764e..b929cbac 100644
--- a/src/lj_str.h
+++ b/src/lj_str.h
@@ -39,12 +39,4 @@ LJ_FUNC GCstr * LJ_FASTCALL lj_str_fromchar(lua_State *L, int c);
39#define LJ_STR_NUMBERBUF LUAI_MAXNUMBER2STR 39#define LJ_STR_NUMBERBUF LUAI_MAXNUMBER2STR
40#define LJ_STR_PTRBUF (2*sizeof(ptrdiff_t)+2) 40#define LJ_STR_PTRBUF (2*sizeof(ptrdiff_t)+2)
41 41
42/* String formatting. */
43LJ_FUNC const char *lj_str_pushvf(lua_State *L, const char *fmt, va_list argp);
44LJ_FUNC const char *lj_str_pushf(lua_State *L, const char *fmt, ...)
45#if defined(__GNUC__)
46 __attribute__ ((format (printf, 2, 3)))
47#endif
48 ;
49
50#endif 42#endif
diff --git a/src/lj_strfmt.c b/src/lj_strfmt.c
index 9aaf08e2..1fe9308d 100644
--- a/src/lj_strfmt.c
+++ b/src/lj_strfmt.c
@@ -10,6 +10,7 @@
10 10
11#include "lj_obj.h" 11#include "lj_obj.h"
12#include "lj_buf.h" 12#include "lj_buf.h"
13#include "lj_state.h"
13#include "lj_char.h" 14#include "lj_char.h"
14#include "lj_strfmt.h" 15#include "lj_strfmt.h"
15 16
@@ -293,3 +294,78 @@ SBuf *lj_strfmt_putnum(SBuf *sb, SFormat sf, lua_Number n)
293 return sb; 294 return sb;
294} 295}
295 296
297/* -- Internal string formatting ------------------------------------------ */
298
299/*
300** These functions are only used for lua_pushfstring(), lua_pushvfstring()
301** and for internal string formatting (e.g. error messages). Caveat: unlike
302** string.format(), only a limited subset of formats and flags are supported!
303**
304** LuaJIT has support for a couple more formats than Lua 5.1/5.2:
305** - %d %u %o %x with full formatting, 32 bit integers only.
306** - %f and other FP formats are really %.14g.
307** - %s %c %p without formatting.
308*/
309
310/* Push formatted message as a string object to Lua stack. va_list variant. */
311const char *lj_strfmt_pushvf(lua_State *L, const char *fmt, va_list argp)
312{
313 SBuf *sb = lj_buf_tmp_(L);
314 FormatState fs;
315 SFormat sf;
316 GCstr *str;
317 lj_strfmt_init(&fs, fmt, strlen(fmt));
318 while ((sf = lj_strfmt_parse(&fs)) != STRFMT_EOF) {
319 switch (STRFMT_TYPE(sf)) {
320 case STRFMT_LIT:
321 lj_buf_putmem(sb, fs.str, fs.len);
322 break;
323 case STRFMT_INT:
324 lj_strfmt_putxint(sb, sf, va_arg(argp, int32_t));
325 break;
326 case STRFMT_UINT:
327 lj_strfmt_putxint(sb, sf, va_arg(argp, uint32_t));
328 break;
329 case STRFMT_NUM: {
330 TValue tv;
331 tv.n = va_arg(argp, lua_Number);
332 setsbufP(sb, lj_str_bufnum(lj_buf_more(sb, LJ_STR_NUMBUF), &tv));
333 break;
334 }
335 case STRFMT_STR: {
336 const char *s = va_arg(argp, char *);
337 if (s == NULL) s = "(null)";
338 lj_buf_putmem(sb, s, (MSize)strlen(s));
339 break;
340 }
341 case STRFMT_CHAR:
342 lj_buf_putb(sb, va_arg(argp, int));
343 break;
344 case STRFMT_PTR:
345 setsbufP(sb, lj_str_bufptr(lj_buf_more(sb, LJ_STR_PTRBUF),
346 va_arg(argp, void *)));
347 break;
348 case STRFMT_ERR:
349 default:
350 lj_buf_putb(sb, '?');
351 lua_assert(0);
352 break;
353 }
354 }
355 str = lj_buf_str(L, sb);
356 setstrV(L, L->top, str);
357 incr_top(L);
358 return strdata(str);
359}
360
361/* Push formatted message as a string object to Lua stack. Vararg variant. */
362const char *lj_strfmt_pushf(lua_State *L, const char *fmt, ...)
363{
364 const char *msg;
365 va_list argp;
366 va_start(argp, fmt);
367 msg = lj_strfmt_pushvf(L, fmt, argp);
368 va_end(argp);
369 return msg;
370}
371
diff --git a/src/lj_strfmt.h b/src/lj_strfmt.h
index b3556f1a..c440f946 100644
--- a/src/lj_strfmt.h
+++ b/src/lj_strfmt.h
@@ -73,6 +73,7 @@ static LJ_AINLINE void lj_strfmt_init(FormatState *fs, const char *p, MSize len)
73} 73}
74 74
75LJ_FUNC SFormat LJ_FASTCALL lj_strfmt_parse(FormatState *fs); 75LJ_FUNC SFormat LJ_FASTCALL lj_strfmt_parse(FormatState *fs);
76
76LJ_FUNC SBuf *lj_strfmt_putchar(SBuf *sb, SFormat, int32_t c); 77LJ_FUNC SBuf *lj_strfmt_putchar(SBuf *sb, SFormat, int32_t c);
77LJ_FUNC SBuf *lj_strfmt_putstr(SBuf *sb, SFormat, GCstr *str); 78LJ_FUNC SBuf *lj_strfmt_putstr(SBuf *sb, SFormat, GCstr *str);
78LJ_FUNC SBuf *lj_strfmt_putquoted(SBuf *sb, GCstr *str); 79LJ_FUNC SBuf *lj_strfmt_putquoted(SBuf *sb, GCstr *str);
@@ -81,4 +82,12 @@ LJ_FUNC SBuf *lj_strfmt_putnum_int(SBuf *sb, SFormat sf, lua_Number n);
81LJ_FUNC SBuf *lj_strfmt_putnum_uint(SBuf *sb, SFormat sf, lua_Number n); 82LJ_FUNC SBuf *lj_strfmt_putnum_uint(SBuf *sb, SFormat sf, lua_Number n);
82LJ_FUNC SBuf *lj_strfmt_putnum(SBuf *sb, SFormat, lua_Number n); 83LJ_FUNC SBuf *lj_strfmt_putnum(SBuf *sb, SFormat, lua_Number n);
83 84
85LJ_FUNC const char *lj_strfmt_pushvf(lua_State *L, const char *fmt,
86 va_list argp);
87LJ_FUNC const char *lj_strfmt_pushf(lua_State *L, const char *fmt, ...)
88#ifdef __GNUC__
89 __attribute__ ((format (printf, 2, 3)))
90#endif
91 ;
92
84#endif 93#endif