* if script not supplied, cancel
*/
-#include "tables.h"
-#include "hbytes.h"
+#include "dgram.h"
typedef struct DgramSocket {
int ix; /* first ! */
int addr_buflen, msg_buflen;
} DgramSocket;
-IdDataTable dgram_socks= { "dgramsock" };
-
-int do_dgram_socket_create(ClientData cd, Tcl_Interp *ip,
+int cht_do_dgramsocket_create(ClientData cd, Tcl_Interp *ip,
SockAddr_Value local, void **sock_r) {
int fd, al, r;
DgramSocket *sock;
const struct sockaddr *sa;
- sa= sockaddr_addr(&local);
- al= sockaddr_len(&local);
+ sa= cht_sockaddr_addr(&local);
+ al= cht_sockaddr_len(&local);
fd= socket(sa->sa_family, SOCK_DGRAM, 0);
- if (fd<0) return posixerr(ip,errno,"socket");
- r= bind(fd, sa, al); if (r) return newfdposixerr(ip,fd,"bind");
- r= setnonblock(fd, 1); if (r) return newfdposixerr(ip,fd,"setnonblock");
+ if (fd<0) return cht_posixerr(ip,errno,"socket");
+ r= bind(fd, sa, al); if (r) return cht_newfdposixerr(ip,fd,"bind");
+ r= cht_setnonblock(fd, 1); if (r) return cht_newfdposixerr(ip,fd,"setnonblock");
sock= TALLOC(sizeof(DgramSocket));
sock->ix= -1;
sock->addr_buf= TALLOC(sock->addr_buflen);
sock->msg_buflen= 0;
sock->msg_buf= 0;
- scriptinv_init(&sock->script);
+ cht_scriptinv_init(&sock->script);
*sock_r= sock;
return TCL_OK;
}
-int do_dgram_socket_transmit(ClientData cd, Tcl_Interp *ip,
+int cht_do_dgramsocket_transmit(ClientData cd, Tcl_Interp *ip,
void *sock_v, HBytes_Value data,
SockAddr_Value remote) {
DgramSocket *sock= sock_v;
int l, r;
r= sendto(sock->fd,
- hbytes_data(&data), l=hbytes_len(&data),
+ cht_hb_data(&data), l=cht_hb_len(&data),
0,
- sockaddr_addr(&remote), sockaddr_len(&remote));
- if (r==-1) return posixerr(ip,errno,"sendto");
- else if (r!=l) return staticerr(ip,"sendto gave wrong answer",0);
+ cht_sockaddr_addr(&remote), cht_sockaddr_len(&remote));
+ if (r==-1) return cht_posixerr(ip,errno,"sendto");
+ else if (r!=l) return cht_staticerr(ip,"sendto gave wrong answer",0);
return TCL_OK;
}
static void cancel(DgramSocket *sock) {
- if (sock->script.obj) {
- scriptinv_cancel(&sock->script);
+ if (sock->script.script) {
+ cht_scriptinv_cancel(&sock->script);
Tcl_DeleteFileHandler(sock->fd);
}
}
static void recv_call(ClientData sock_cd, int mask) {
DgramSocket *sock= (void*)sock_cd;
- Tcl_Interp *ip= sock->script.ip;
+ Tcl_Interp *ip= sock->script.ipq;
int sz, rc, peek;
HBytes_Value message_val;
SockAddr_Value peer_val;
struct msghdr mh;
struct iovec iov;
- hbytes_empty(&message_val);
- sockaddr_clear(&peer_val);
+ cht_hb_empty(&message_val);
+ cht_sockaddr_clear(&peer_val);
mh.msg_iov= &iov;
mh.msg_iovlen= 1;
sz= recvmsg(sock->fd, &mh, peek);
if (sz==-1) {
if (errno == EAGAIN || errno == EWOULDBLOCK) rc=0;
- else rc= posixerr(ip,errno,"recvmsg");
+ else rc= cht_posixerr(ip,errno,"recvmsg");
goto x_rc;
}
}
TFREE(sock->msg_buf);
+ assert(sock->msg_buflen < INT_MAX/4);
sock->msg_buflen *= 2;
sock->msg_buflen += 100;
sock->msg_buf= TALLOC(sock->msg_buflen);
}
- hbytes_array(&message_val, iov.iov_base, sz);
- sockaddr_create(&peer_val, mh.msg_name, mh.msg_namelen);
+ cht_hb_array(&message_val, iov.iov_base, sz);
+ cht_sockaddr_create(&peer_val, mh.msg_name, mh.msg_namelen);
- args[0]= ret_hb(ip, message_val); hbytes_empty(&message_val);
- args[1]= ret_sockaddr(ip, peer_val); sockaddr_clear(&peer_val);
- args[2]= ret_iddata(ip, sock, &dgram_socks);
- scriptinv_invoke(&sock->script,3,args);
+ args[0]= cht_ret_hb(ip, message_val); cht_hb_empty(&message_val);
+ args[1]= cht_ret_sockaddr(ip, peer_val); cht_sockaddr_clear(&peer_val);
+ args[2]= cht_ret_iddata(ip, sock, &cht_dgram_socks);
+ cht_scriptinv_invoke(&sock->script,3,args);
rc= 0;
Tcl_BackgroundError(ip);
}
-int do_dgram_socket_on_receive(ClientData cd, Tcl_Interp *ip,
+int cht_do_dgramsocket_on_receive(ClientData cd, Tcl_Interp *ip,
void *sock_v, Tcl_Obj *newscript) {
DgramSocket *sock= sock_v;
int rc;
cancel(sock);
if (newscript) {
- rc= scriptinv_set(&sock->script, ip, newscript);
+ rc= cht_scriptinv_set(&sock->script, ip, newscript, 0);
if (rc) return rc;
}
return TCL_OK;
}
-int do_dgram_socket_close(ClientData cd, Tcl_Interp *ip, void *sock_v) {
- DgramSocket *sock= sock_v;
-
+static void destroy(DgramSocket *sock) {
cancel(sock);
close(sock->fd); /* nothing useful to be done with errors */
- tabledataid_disposing(sock,&dgram_socks);
TFREE(sock->addr_buf);
TFREE(sock->msg_buf);
TFREE(sock);
+}
+
+static void destroy_idtabcb(Tcl_Interp *ip, void *sock_v) {
+ destroy(sock_v);
+}
+
+int cht_do_dgramsocket_close(ClientData cd, Tcl_Interp *ip, void *sock_v) {
+ cht_tabledataid_disposing(ip,sock_v,&cht_dgram_socks);
+ destroy(sock_v);
return TCL_OK;
}
+
+const IdDataSpec cht_dgram_socks= {
+ "dgramsock", "dgramsock-table", destroy_idtabcb
+};