sq_rs232.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. /*
  2. * Copyright (c) 2011 Petr Stetiar <[email protected]>, Gaben Ltd.
  3. *
  4. * Permission is hereby granted, free of charge, to any person
  5. * obtaining a copy of this software and associated documentation
  6. * files (the "Software"), to deal in the Software without
  7. * restriction, including without limitation the rights to use,
  8. * copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. * copies of the Software, and to permit persons to whom the
  10. * Software is furnished to do so, subject to the following
  11. * conditions:
  12. *
  13. * The above copyright notice and this permission notice shall be
  14. * included in all copies or substantial portions of the Software.
  15. *
  16. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  17. * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
  18. * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  19. * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
  20. * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  21. * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  22. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  23. * OTHER DEALINGS IN THE SOFTWARE.
  24. *
  25. * Ported to Squirrel/SquiLu by Domingo Alvarez Duarte
  26. */
  27. #include "squirrel.h"
  28. #include <string.h>
  29. #include <string.h>
  30. #include <stdlib.h>
  31. SQ_OPT_STRING_STRLEN();
  32. #include "librs232/rs232.h"
  33. #include "librs232/rs232.c"
  34. #ifdef WIN32
  35. #include "librs232/rs232_windows.h"
  36. #include "librs232/rs232_windows.c"
  37. #else
  38. #include "librs232/rs232_posix.h"
  39. #include "librs232/rs232_posix.c"
  40. #endif
  41. #define MODULE_TIMESTAMP __DATE__ " " __TIME__
  42. #define MODULE_VERSION "1.0.3"
  43. #define MODULE_BUILD "$Id: luars232.c 15 2011-02-23 09:02:20Z sp $"
  44. #define MODULE_COPYRIGHT "Copyright (c) 2011 Petr Stetiar <[email protected]>, Gaben Ltd."
  45. static const SQChar SQRS232_TAG[] = _SC("SqRs232");
  46. static SQRESULT get_rs232_instance(HSQUIRRELVM v, SQInteger idx, struct rs232_port_t **self){
  47. SQRESULT _rc_;
  48. if((_rc_ = sq_getinstanceup(v,idx,(SQUserPointer*)self,(void*)SQRS232_TAG,SQFalse)) < 0) return _rc_;
  49. if(!*self) return sq_throwerror(v, _SC("rs232 is closed"));
  50. return _rc_;
  51. }
  52. #define GET_sq_rs232_INSTANCE(v, idx) struct rs232_port_t *self=NULL; \
  53. if((_rc_ = get_rs232_instance(v,idx,&self)) < 0) return _rc_;
  54. static SQRESULT sq_rs232_release_hook(SQUserPointer p, SQInteger size, void *ep) {
  55. struct rs232_port_t *self = (struct rs232_port_t *)p;
  56. if(self) rs232_end(self);
  57. return 0;
  58. }
  59. /*
  60. ** Creates a new SqRs232.
  61. */
  62. static SQRESULT sq_rs232_constructor (HSQUIRRELVM v) {
  63. sq_setinstanceup(v, 1, 0);
  64. sq_setreleasehook(v, 1, sq_rs232_release_hook);
  65. return 1;
  66. }
  67. static SQRESULT sq_rs232_open(HSQUIRRELVM v){
  68. SQ_FUNC_VARS_NO_TOP(v);
  69. struct rs232_port_t *self=NULL;
  70. if((_rc_ = sq_getinstanceup(v,1,(SQUserPointer*)&self,(void*)SQRS232_TAG,SQFalse)) < 0) return _rc_;
  71. SQ_GET_STRING(v, 2, device);
  72. if(self != NULL) sq_throwerror(v, _SC("rs232 already opened"));
  73. self = rs232_init();
  74. if(self == NULL) sq_throwerror(v, _SC("%s"), rs232_strerror(RS232_ERR_CONFIG));
  75. rs232_set_device(self, (char*)device);
  76. int ret = rs232_open(self);
  77. if (ret > RS232_ERR_NOERROR) {
  78. free(self->pt);
  79. free(self);
  80. return sq_throwerror(v, _SC("%s"), rs232_strerror(ret));
  81. }
  82. sq_setinstanceup(v, 1, self);
  83. return 0;
  84. }
  85. static SQRESULT sq_rs232_close(HSQUIRRELVM v){
  86. SQ_FUNC_VARS_NO_TOP(v);
  87. GET_sq_rs232_INSTANCE(v, 1);
  88. _rc_ = sq_rs232_release_hook(self, 0, v);
  89. sq_setinstanceup(v, 1, 0); //next calls will fail with "rs232 port already closed"
  90. sq_pushinteger(v, _rc_);
  91. return 1;
  92. }
  93. static SQRESULT sq_rs232_read(HSQUIRRELVM v){
  94. SQ_FUNC_VARS(v);
  95. GET_sq_rs232_INSTANCE(v, 1);
  96. SQ_GET_INTEGER(v, 2, len);
  97. unsigned int bytes_read = 0;
  98. SQChar *data = NULL;
  99. switch (_top_) {
  100. case 2:{
  101. data = sq_getscratchpad(v, len);
  102. _rc_ = rs232_read(self, (unsigned char *)data, len, &bytes_read);
  103. }
  104. break;
  105. case 3:
  106. case 4:{
  107. SQ_GET_INTEGER(v, 3, timeout);
  108. SQ_OPT_BOOL(v, 4, forced, SQFalse);
  109. data = sq_getscratchpad(v, len);
  110. if(forced) _rc_ = rs232_read_timeout_forced(self, (unsigned char *)data, len, &bytes_read, timeout);
  111. else _rc_ = rs232_read_timeout(self, (unsigned char *)data, len, &bytes_read, timeout);
  112. }
  113. break;
  114. default:
  115. return sq_throwerror(v, _SC("wrong number of parameters " _PRINT_INT_FMT), _top_-1);
  116. }
  117. DBG("rc=%d hex='%s' bytes_read=%d\n",
  118. _rc_, rs232_hex_dump(data, bytes_read), bytes_read);
  119. if(bytes_read > 0) sq_pushstring(v, data, bytes_read);
  120. else sq_pushinteger(v, _rc_);
  121. return 1;
  122. }
  123. static SQRESULT sq_rs232_write(HSQUIRRELVM v){
  124. SQ_FUNC_VARS(v);
  125. GET_sq_rs232_INSTANCE(v, 1);
  126. SQ_GET_STRING(v, 2, data);
  127. unsigned int wlen = 0;
  128. switch (_top_) {
  129. case 2:{
  130. _rc_ = rs232_write(self, (unsigned char*) data, (unsigned int) data_size, &wlen);
  131. }
  132. break;
  133. case 3:{
  134. SQ_GET_INTEGER(v, 3, timeout);
  135. _rc_ = rs232_write_timeout(self, (unsigned char*) data, (unsigned int) data_size, &wlen, timeout);
  136. }
  137. break;
  138. default:
  139. return sq_throwerror(v, _SC("wrong number of parameters " _PRINT_INT_FMT), _top_-1);
  140. }
  141. if(_rc_ > RS232_ERR_NOERROR) return sq_throwerror(v, _SC("%s"), rs232_strerror(_rc_));
  142. sq_pushinteger(v, wlen);
  143. return 1;
  144. }
  145. static SQRESULT sq_rs232_in_queue(HSQUIRRELVM v){
  146. SQ_FUNC_VARS_NO_TOP(v);
  147. GET_sq_rs232_INSTANCE(v, 1);
  148. unsigned int in_bytes;
  149. int rc = rs232_in_queue(self, &in_bytes);
  150. if(rc > RS232_ERR_NOERROR) return sq_throwerror(v, _SC("%s"), rs232_strerror(rc));
  151. sq_pushinteger(v, in_bytes);
  152. return 1;
  153. }
  154. static SQRESULT sq_rs232_in_queue_clear(HSQUIRRELVM v){
  155. SQ_FUNC_VARS_NO_TOP(v);
  156. GET_sq_rs232_INSTANCE(v, 1);
  157. rs232_in_queue_clear(self);
  158. return 0;
  159. }
  160. static SQRESULT sq_rs232_flush(HSQUIRRELVM v){
  161. SQ_FUNC_VARS_NO_TOP(v);
  162. GET_sq_rs232_INSTANCE(v, 1);
  163. sq_pushinteger(v, rs232_flush(self));
  164. return 1;
  165. }
  166. static SQRESULT sq_rs232_device(HSQUIRRELVM v){
  167. SQ_FUNC_VARS_NO_TOP(v);
  168. GET_sq_rs232_INSTANCE(v, 1);
  169. const char *ret = rs232_get_device(self);
  170. if (ret == NULL) sq_pushnull(v);
  171. else sq_pushstring(v, ret, -1);
  172. return 1;
  173. }
  174. static SQRESULT sq_rs232_fd(HSQUIRRELVM v){
  175. SQ_FUNC_VARS_NO_TOP(v);
  176. GET_sq_rs232_INSTANCE(v, 1);
  177. sq_pushinteger(v, rs232_fd(self));
  178. return 1;
  179. }
  180. static SQRESULT sq_rs232_strerror(HSQUIRRELVM v){
  181. SQ_FUNC_VARS_NO_TOP(v);
  182. SQ_GET_INTEGER(v, 2, error_code);
  183. sq_pushstring(v, rs232_strerror(error_code), -1);
  184. return 1;
  185. }
  186. static SQRESULT sq_rs232__tostring(HSQUIRRELVM v){
  187. SQ_FUNC_VARS_NO_TOP(v);
  188. GET_sq_rs232_INSTANCE(v, 1);
  189. sq_pushstring(v, rs232_to_string(self), -1);
  190. return 1;
  191. }
  192. #define FN_GET_SET_PORT(type, sufix) \
  193. static SQRESULT sq_rs232_##type##sufix(HSQUIRRELVM v) \
  194. { \
  195. SQ_FUNC_VARS(v);\
  196. GET_sq_rs232_INSTANCE(v, 1);\
  197. if(_top_ > 1) {\
  198. SQ_GET_INTEGER(v, 2, iparam);\
  199. sq_pushinteger(v, rs232_set_##type(self, (unsigned int)iparam));\
  200. }\
  201. else sq_pushinteger(v, rs232_get_##type(self));\
  202. return 1;\
  203. }
  204. #define FN_GET_PORT_STRING(type, sufix) \
  205. static SQRESULT sq_rs232_##type##sufix##_tostring(HSQUIRRELVM v) \
  206. { \
  207. SQ_FUNC_VARS(v);\
  208. GET_sq_rs232_INSTANCE(v, 1);\
  209. SQ_OPT_INTEGER(v, 2, iparam, -1);\
  210. const char *ret = rs232_str##type(iparam == -1 ? rs232_get_##type(self) : (unsigned int) iparam); \
  211. if (ret == NULL) sq_pushnull(v); \
  212. else sq_pushstring(v, ret, -1); \
  213. return 1; \
  214. }
  215. #define FN_GET_SET_STR_PORT(type, sufix) FN_GET_SET_PORT(type, sufix); FN_GET_PORT_STRING(type, sufix)
  216. FN_GET_SET_STR_PORT(baud, _rate);
  217. FN_GET_SET_STR_PORT(data, _bits);
  218. FN_GET_SET_STR_PORT(stop, _bits);
  219. FN_GET_SET_STR_PORT(parity,);
  220. FN_GET_SET_STR_PORT(flow, _control);
  221. FN_GET_SET_STR_PORT(dtr,);
  222. FN_GET_SET_STR_PORT(rts,);
  223. #define _DECL_FUNC(name,nparams,tycheck) {_SC(#name), sq_rs232_##name,nparams,tycheck}
  224. static SQRegFunction rs232_methods[] =
  225. {
  226. _DECL_FUNC(constructor,1,_SC("x")),
  227. _DECL_FUNC(open,2,_SC("xs")),
  228. _DECL_FUNC(close,1,_SC("x")),
  229. _DECL_FUNC(read,-2,_SC("xiib")),
  230. _DECL_FUNC(write,-2,_SC("xsi")),
  231. _DECL_FUNC(flush,1,_SC("x")),
  232. _DECL_FUNC(in_queue,1,_SC("x")),
  233. _DECL_FUNC(in_queue_clear,1,_SC("x")),
  234. _DECL_FUNC(device,1,_SC("x")),
  235. _DECL_FUNC(fd,1,_SC("x")),
  236. _DECL_FUNC(strerror,2,_SC(".i")),
  237. _DECL_FUNC(_tostring,1,_SC("x")),
  238. /* baud */
  239. _DECL_FUNC(baud_rate,-1,_SC("xi")),
  240. _DECL_FUNC(baud_rate_tostring,-1,_SC(".i")),
  241. /* data */
  242. _DECL_FUNC(data_bits,-1,_SC("xi")),
  243. _DECL_FUNC(data_bits_tostring,-1,_SC(".i")),
  244. /* stop */
  245. _DECL_FUNC(stop_bits,-1,_SC("xi")),
  246. _DECL_FUNC(stop_bits_tostring,-1,_SC(".i")),
  247. /* parity */
  248. _DECL_FUNC(parity,-1,_SC("xi")),
  249. _DECL_FUNC(parity_tostring,-1,_SC(".i")),
  250. /* flow */
  251. _DECL_FUNC(flow_control,-1,_SC("xi")),
  252. _DECL_FUNC(flow_control_tostring,-1,_SC(".i")),
  253. /* dtr */
  254. _DECL_FUNC(dtr,-1,_SC("xi")),
  255. _DECL_FUNC(dtr_tostring,-1,_SC(".i")),
  256. /* rts */
  257. _DECL_FUNC(rts,-1,_SC("xi")),
  258. _DECL_FUNC(rts_tostring,-1,_SC(".i")),
  259. {0,0}
  260. };
  261. #define INT_CONST(v,num) sq_pushstring(v,_SC(#num),-1);sq_pushinteger(v,num);sq_newslot(v,-3,SQTrue);
  262. SQRESULT sqext_register_rs232(HSQUIRRELVM v)
  263. {
  264. sq_pushstring(v,SQRS232_TAG,-1);
  265. sq_newclass(v,SQFalse);
  266. sq_settypetag(v,-1,(void*)SQRS232_TAG);
  267. sq_insert_reg_funcs(v, rs232_methods);
  268. INT_CONST(v, RS232_BAUD_300 );
  269. INT_CONST(v, RS232_BAUD_2400 );
  270. INT_CONST(v, RS232_BAUD_4800 );
  271. INT_CONST(v, RS232_BAUD_9600 );
  272. INT_CONST(v, RS232_BAUD_19200 );
  273. INT_CONST(v, RS232_BAUD_38400 );
  274. INT_CONST(v, RS232_BAUD_57600 );
  275. INT_CONST(v, RS232_BAUD_115200 );
  276. INT_CONST(v, RS232_BAUD_460800 );
  277. /* databits */
  278. INT_CONST(v, RS232_DATA_5 );
  279. INT_CONST(v, RS232_DATA_6 );
  280. INT_CONST(v, RS232_DATA_7 );
  281. INT_CONST(v, RS232_DATA_8 );
  282. /* stop bits */
  283. INT_CONST(v, RS232_STOP_1 );
  284. INT_CONST(v, RS232_STOP_2 );
  285. /* parity */
  286. INT_CONST(v, RS232_PARITY_NONE );
  287. INT_CONST(v, RS232_PARITY_ODD );
  288. INT_CONST(v, RS232_PARITY_EVEN );
  289. /* flow */
  290. INT_CONST(v, RS232_FLOW_OFF );
  291. INT_CONST(v, RS232_FLOW_HW );
  292. INT_CONST(v, RS232_FLOW_XON_XOFF );
  293. /* DTR and RTS */
  294. INT_CONST(v, RS232_DTR_ON );
  295. INT_CONST(v, RS232_DTR_OFF );
  296. INT_CONST(v, RS232_RTS_ON );
  297. INT_CONST(v, RS232_RTS_OFF );
  298. /* errors */
  299. INT_CONST(v, RS232_ERR_NOERROR );
  300. INT_CONST(v, RS232_ERR_UNKNOWN );
  301. INT_CONST(v, RS232_ERR_OPEN );
  302. INT_CONST(v, RS232_ERR_CLOSE );
  303. INT_CONST(v, RS232_ERR_FLUSH );
  304. INT_CONST(v, RS232_ERR_CONFIG );
  305. INT_CONST(v, RS232_ERR_READ );
  306. INT_CONST(v, RS232_ERR_WRITE );
  307. INT_CONST(v, RS232_ERR_SELECT );
  308. INT_CONST(v, RS232_ERR_TIMEOUT );
  309. INT_CONST(v, RS232_ERR_IOCTL );
  310. INT_CONST(v, RS232_ERR_PORT_CLOSED );
  311. sq_newslot(v,-3,SQTrue);
  312. return 1;
  313. }