sqstdstream.cpp 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  1. /* see copyright notice in squirrel.h */
  2. #include <new>
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include <squirrel.h>
  7. #include <sqstdio.h>
  8. #include <sqstdblob.h>
  9. #include "sqstdstream.h"
  10. #include "sqstdblobimpl.h"
  11. #define SETUP_STREAM(v) \
  12. SQStream *self = NULL; \
  13. if(SQ_FAILED(sq_getinstanceup(v,1,(SQUserPointer*)&self,(SQUserPointer)SQSTD_STREAM_TYPE_TAG))) \
  14. return sq_throwerror(v,_SC("invalid type tag")); \
  15. if(!self || !self->IsValid()) \
  16. return sq_throwerror(v,_SC("the stream is invalid"));
  17. SQInteger _stream_read_line(HSQUIRRELVM v) {
  18. SETUP_STREAM(v);
  19. SQBlob b(0,BLOB_BUFSIZE);
  20. char c;
  21. int len = sizeof(char);
  22. while(!self->EOS()){
  23. if(self->Read(&c, len) != len){
  24. if(!self->EOS()) return sq_throwerror(v,_SC("io error"));
  25. c = '\n';
  26. }
  27. if(c == '\n') break;
  28. b.Write(&c, sizeof(char));
  29. }
  30. if(b.Len() > 0) sq_pushstring(v, (const SQChar*)b.GetBuf(), b.Len());
  31. else sq_pushnull(v);
  32. return 1;
  33. }
  34. SQInteger _stream_read(HSQUIRRELVM v)
  35. {
  36. SETUP_STREAM(v);
  37. SQChar *data;
  38. SQInteger size,res;
  39. sq_getinteger(v,2,&size);
  40. if(size > self->Len()) {
  41. size = self->Len();
  42. }
  43. data = sq_getscratchpad(v,size);
  44. res = self->Read(data,size);
  45. if(res <= 0)
  46. return sq_throwerror(v,_SC("no data left to read"));
  47. sq_pushstring(v,data,res);
  48. return 1;
  49. }
  50. SQInteger _stream_readblob(HSQUIRRELVM v)
  51. {
  52. SETUP_STREAM(v);
  53. SQUserPointer data,blobp;
  54. SQInteger size,res;
  55. sq_getinteger(v,2,&size);
  56. if(size > self->Len()) {
  57. size = self->Len();
  58. }
  59. data = sq_getscratchpad(v,size);
  60. res = self->Read(data,size);
  61. if(res <= 0)
  62. return sq_throwerror(v,_SC("no data left to read"));
  63. blobp = sqstd_createblob(v,res);
  64. memcpy(blobp,data,res);
  65. return 1;
  66. }
  67. #define SAFE_READN(ptr,len) { \
  68. if(self->Read(ptr,len) != len) return sq_throwerror(v,_SC("io error")); \
  69. }
  70. SQInteger _stream_readn(HSQUIRRELVM v)
  71. {
  72. SETUP_STREAM(v);
  73. SQInteger format;
  74. sq_getinteger(v, 2, &format);
  75. switch(format) {
  76. case 'l': {
  77. SQInteger i;
  78. SAFE_READN(&i, sizeof(i));
  79. sq_pushinteger(v, i);
  80. }
  81. break;
  82. case 'i': {
  83. SQInt32 i;
  84. SAFE_READN(&i, sizeof(i));
  85. sq_pushinteger(v, i);
  86. }
  87. break;
  88. case 's': {
  89. short s;
  90. SAFE_READN(&s, sizeof(short));
  91. sq_pushinteger(v, s);
  92. }
  93. break;
  94. case 'w': {
  95. unsigned short w;
  96. SAFE_READN(&w, sizeof(unsigned short));
  97. sq_pushinteger(v, w);
  98. }
  99. break;
  100. case 'c': {
  101. char c;
  102. SAFE_READN(&c, sizeof(char));
  103. sq_pushinteger(v, c);
  104. }
  105. break;
  106. case 'b': {
  107. unsigned char c;
  108. SAFE_READN(&c, sizeof(unsigned char));
  109. sq_pushinteger(v, c);
  110. }
  111. break;
  112. case 'f': {
  113. float f;
  114. SAFE_READN(&f, sizeof(float));
  115. sq_pushfloat(v, f);
  116. }
  117. break;
  118. case 'd': {
  119. double d;
  120. SAFE_READN(&d, sizeof(double));
  121. sq_pushfloat(v, (SQFloat)d);
  122. }
  123. break;
  124. default:
  125. return sq_throwerror(v, _SC("invalid format"));
  126. }
  127. return 1;
  128. }
  129. SQInteger _stream_write(HSQUIRRELVM v)
  130. {
  131. const SQChar *str;
  132. SQInteger total_size, size;
  133. SETUP_STREAM(v);
  134. total_size = 0;
  135. for(SQInteger i=2, len=sq_gettop(v); i <= len; ++i){
  136. if(SQ_FAILED(sq_tostring(v,i)))
  137. return sq_throwerror(v,_SC("invalid parameter"));
  138. sq_getstring(v,-1,&str);
  139. size = sq_getsize(v,-1);
  140. if(self->Write((SQChar*)str,size) != size)
  141. return sq_throwerror(v,_SC("io error"));
  142. sq_poptop(v); //remove converted string
  143. total_size += size;
  144. }
  145. sq_pushinteger(v,total_size);
  146. return 1;
  147. }
  148. SQInteger _stream_write_fmt(HSQUIRRELVM v)
  149. {
  150. const SQChar *str;
  151. SQInteger size;
  152. SETUP_STREAM(v);
  153. if(SQ_FAILED(sq_tostring(v,2)))
  154. return sq_throwerror(v,_SC("invalid parameter"));
  155. sq_getstring(v,-1,&str);
  156. size = sq_getsize(v,-1);
  157. if(self->Write((SQChar*)str,size) != size)
  158. return sq_throwerror(v,_SC("io error"));
  159. sq_poptop(v); //remove converted string
  160. sq_pushinteger(v,size);
  161. return 1;
  162. }
  163. SQInteger _stream_writeblob(HSQUIRRELVM v)
  164. {
  165. SQUserPointer data;
  166. SQInteger size;
  167. SETUP_STREAM(v);
  168. if(SQ_FAILED(sqstd_getblob(v,2,&data)))
  169. return sq_throwerror(v,_SC("invalid parameter"));
  170. size = sqstd_getblobsize(v,2);
  171. if(self->Write(data,size) != size)
  172. return sq_throwerror(v,_SC("io error"));
  173. sq_pushinteger(v,size);
  174. return 1;
  175. }
  176. SQInteger _stream_writen(HSQUIRRELVM v)
  177. {
  178. SETUP_STREAM(v);
  179. SQInteger format, ti;
  180. SQFloat tf;
  181. sq_getinteger(v, 3, &format);
  182. switch(format) {
  183. case 'l': {
  184. SQInteger i;
  185. sq_getinteger(v, 2, &ti);
  186. i = ti;
  187. self->Write(&i, sizeof(SQInteger));
  188. }
  189. break;
  190. case 'i': {
  191. SQInt32 i;
  192. sq_getinteger(v, 2, &ti);
  193. i = (SQInt32)ti;
  194. self->Write(&i, sizeof(SQInt32));
  195. }
  196. break;
  197. case 's': {
  198. short s;
  199. sq_getinteger(v, 2, &ti);
  200. s = (short)ti;
  201. self->Write(&s, sizeof(short));
  202. }
  203. break;
  204. case 'w': {
  205. unsigned short w;
  206. sq_getinteger(v, 2, &ti);
  207. w = (unsigned short)ti;
  208. self->Write(&w, sizeof(unsigned short));
  209. }
  210. break;
  211. case 'c': {
  212. char c;
  213. sq_getinteger(v, 2, &ti);
  214. c = (char)ti;
  215. self->Write(&c, sizeof(char));
  216. }
  217. break;
  218. case 'b': {
  219. unsigned char b;
  220. sq_getinteger(v, 2, &ti);
  221. b = (unsigned char)ti;
  222. self->Write(&b, sizeof(unsigned char));
  223. }
  224. break;
  225. case 'f': {
  226. float f;
  227. sq_getfloat(v, 2, &tf);
  228. f = (float)tf;
  229. self->Write(&f, sizeof(float));
  230. }
  231. break;
  232. case 'd': {
  233. double d;
  234. sq_getfloat(v, 2, &tf);
  235. d = tf;
  236. self->Write(&d, sizeof(double));
  237. }
  238. break;
  239. default:
  240. return sq_throwerror(v, _SC("invalid format"));
  241. }
  242. return 0;
  243. }
  244. SQInteger _stream_seek(HSQUIRRELVM v)
  245. {
  246. SETUP_STREAM(v);
  247. SQInteger offset, origin = SQ_SEEK_SET;
  248. sq_getinteger(v, 2, &offset);
  249. if(sq_gettop(v) > 2) {
  250. SQInteger t;
  251. sq_getinteger(v, 3, &t);
  252. switch(t) {
  253. case 'b': origin = SQ_SEEK_SET; break;
  254. case 'c': origin = SQ_SEEK_CUR; break;
  255. case 'e': origin = SQ_SEEK_END; break;
  256. default: return sq_throwerror(v,_SC("invalid origin"));
  257. }
  258. }
  259. sq_pushinteger(v, self->Seek(offset, origin));
  260. return 1;
  261. }
  262. SQInteger _stream_tell(HSQUIRRELVM v)
  263. {
  264. SETUP_STREAM(v);
  265. sq_pushinteger(v, self->Tell());
  266. return 1;
  267. }
  268. SQInteger _stream_len(HSQUIRRELVM v)
  269. {
  270. SETUP_STREAM(v);
  271. sq_pushinteger(v, self->Len());
  272. return 1;
  273. }
  274. SQInteger _stream_flush(HSQUIRRELVM v)
  275. {
  276. SETUP_STREAM(v);
  277. if(!self->Flush())
  278. sq_pushinteger(v, 1);
  279. else
  280. sq_pushnull(v);
  281. return 1;
  282. }
  283. SQInteger _stream_eos(HSQUIRRELVM v)
  284. {
  285. SETUP_STREAM(v);
  286. if(self->EOS())
  287. sq_pushinteger(v, 1);
  288. else
  289. sq_pushnull(v);
  290. return 1;
  291. }
  292. SQInteger _stream__cloned(HSQUIRRELVM v)
  293. {
  294. return sq_throwerror(v,_SC("this object cannot be cloned"));
  295. }
  296. static SQRegFunction _stream_methods[] = {
  297. _DECL_STREAM_FUNC(read_line,1,_SC("x")),
  298. _DECL_STREAM_FUNC(read,2,_SC("xn")),
  299. _DECL_STREAM_FUNC(readblob,2,_SC("xn")),
  300. _DECL_STREAM_FUNC(readn,2,_SC("xn")),
  301. _DECL_STREAM_FUNC(write,-2,_SC("x.")),
  302. _DECL_STREAM_FUNC(writeblob,-2,_SC("xx")),
  303. _DECL_STREAM_FUNC(writen,3,_SC("xnn")),
  304. _DECL_STREAM_FUNC(seek,-2,_SC("xnn")),
  305. _DECL_STREAM_FUNC(tell,1,_SC("x")),
  306. _DECL_STREAM_FUNC(len,1,_SC("x")),
  307. _DECL_STREAM_FUNC(eos,1,_SC("x")),
  308. _DECL_STREAM_FUNC(flush,1,_SC("x")),
  309. _DECL_STREAM_FUNC(_cloned,0,NULL),
  310. {0,0}
  311. };
  312. void init_streamclass(HSQUIRRELVM v)
  313. {
  314. sq_pushregistrytable(v);
  315. sq_pushstring(v,_SC("std_stream"),-1);
  316. if(SQ_FAILED(sq_get(v,-2))) {
  317. sq_pushstring(v,_SC("std_stream"),-1);
  318. sq_newclass(v,SQFalse);
  319. sq_settypetag(v,-1,(SQUserPointer)SQSTD_STREAM_TYPE_TAG);
  320. SQInteger i = 0;
  321. while(_stream_methods[i].name != 0) {
  322. SQRegFunction &f = _stream_methods[i];
  323. sq_pushstring(v,f.name,-1);
  324. sq_newclosure(v,f.f,0);
  325. sq_setparamscheck(v,f.nparamscheck,f.typemask);
  326. sq_newslot(v,-3,SQFalse);
  327. i++;
  328. }
  329. sq_newslot(v,-3,SQFalse);
  330. sq_pushroottable(v);
  331. sq_pushstring(v,_SC("stream"),-1);
  332. sq_pushstring(v,_SC("std_stream"),-1);
  333. sq_get(v,-4);
  334. sq_newslot(v,-3,SQFalse);
  335. sq_pop(v,1);
  336. }
  337. else {
  338. sq_pop(v,1); //result
  339. }
  340. sq_pop(v,1);
  341. }
  342. SQRESULT declare_stream(HSQUIRRELVM v,const SQChar* name,SQUserPointer typetag,const SQChar* reg_name,SQRegFunction *methods,SQRegFunction *globals)
  343. {
  344. if(sq_gettype(v,-1) != OT_TABLE)
  345. return sq_throwerror(v,_SC("table expected"));
  346. SQInteger top = sq_gettop(v);
  347. //create delegate
  348. init_streamclass(v);
  349. sq_pushregistrytable(v);
  350. sq_pushstring(v,reg_name,-1);
  351. sq_pushstring(v,_SC("std_stream"),-1);
  352. if(SQ_SUCCEEDED(sq_get(v,-3))) {
  353. sq_newclass(v,SQTrue);
  354. sq_settypetag(v,-1,typetag);
  355. SQInteger i = 0;
  356. while(methods[i].name != 0) {
  357. SQRegFunction &f = methods[i];
  358. sq_pushstring(v,f.name,-1);
  359. sq_newclosure(v,f.f,0);
  360. sq_setparamscheck(v,f.nparamscheck,f.typemask);
  361. sq_setnativeclosurename(v,-1,f.name);
  362. sq_newslot(v,-3,SQFalse);
  363. i++;
  364. }
  365. sq_newslot(v,-3,SQFalse);
  366. sq_pop(v,1);
  367. i = 0;
  368. while(globals[i].name!=0)
  369. {
  370. SQRegFunction &f = globals[i];
  371. sq_pushstring(v,f.name,-1);
  372. sq_newclosure(v,f.f,0);
  373. sq_setparamscheck(v,f.nparamscheck,f.typemask);
  374. sq_setnativeclosurename(v,-1,f.name);
  375. sq_newslot(v,-3,SQFalse);
  376. i++;
  377. }
  378. //register the class in the target table
  379. sq_pushstring(v,name,-1);
  380. sq_pushregistrytable(v);
  381. sq_pushstring(v,reg_name,-1);
  382. sq_get(v,-2);
  383. sq_remove(v,-2);
  384. sq_newslot(v,-3,SQFalse);
  385. sq_settop(v,top);
  386. return SQ_OK;
  387. }
  388. sq_settop(v,top);
  389. return SQ_ERROR;
  390. }