/hacked-cadmium/src/main/java/fr/x9c/cadmium/primitives/unix/Sendrecv.java

http://jhmtasc.googlecode.com/ · Java · 306 lines · 219 code · 12 blank · 75 comment · 24 complexity · 319e3cba172fbb4035f6c548f574f852 MD5 · raw file

  1. /*
  2. * This file is part of Cadmium.
  3. * Copyright (C) 2007-2010 Xavier Clerc.
  4. *
  5. * Cadmium is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU Lesser General Public License as published by
  7. * the Free Software Foundation; either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * Cadmium is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU Lesser General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU Lesser General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. package fr.x9c.cadmium.primitives.unix;
  19. import java.io.IOException;
  20. import java.io.InterruptedIOException;
  21. import java.net.DatagramPacket;
  22. import java.net.DatagramSocket;
  23. import java.net.InetAddress;
  24. import java.net.InetSocketAddress;
  25. import java.net.SocketAddress;
  26. import fr.x9c.cadmium.kernel.Block;
  27. import fr.x9c.cadmium.kernel.Channel;
  28. import fr.x9c.cadmium.kernel.CodeRunner;
  29. import fr.x9c.cadmium.kernel.Context;
  30. import fr.x9c.cadmium.kernel.Fail;
  31. import fr.x9c.cadmium.kernel.FalseExit;
  32. import fr.x9c.cadmium.kernel.Primitive;
  33. import fr.x9c.cadmium.kernel.PrimitiveProvider;
  34. import fr.x9c.cadmium.kernel.Value;
  35. /**
  36. * This class provides implementation for socket i/o-related primitives.
  37. *
  38. * @author <a href="mailto:cadmium@x9c.fr">Xavier Clerc</a>
  39. * @version 1.0
  40. * @since 1.0
  41. */
  42. @PrimitiveProvider
  43. public final class Sendrecv {
  44. /**
  45. * No instance of this class.
  46. */
  47. private Sendrecv() {
  48. } // end empty constructor
  49. /**
  50. * Receives a packet on a connected socket.
  51. * @param ctxt context
  52. * @param socket socket
  53. * @param buff buffer of data to receive
  54. * @param ofs offset of data to receive
  55. * @param len maximum length of data to receive
  56. * @param flags ignored
  57. * @return number of bytes received
  58. * @throws Fail.Exception if an i/o error occurs
  59. */
  60. @Primitive
  61. public static Value unix_recv(final CodeRunner ctxt,
  62. final Value socket,
  63. final Value buff,
  64. final Value ofs,
  65. final Value len,
  66. final Value flags)
  67. throws Fail.Exception, FalseExit {
  68. final Context context = ctxt.getContext();
  69. try {
  70. final Channel ch = context.getChannel(socket.asLong());
  71. if (ch == null) {
  72. Unix.fail(ctxt, "recv", Unix.INVALID_DESCRIPTOR_MSG);
  73. return Value.UNIT; // never reached
  74. } // end if
  75. final java.net.Socket s = ch.asSocket();
  76. final DatagramSocket ds = ch.asDatagramSocket();
  77. if (s != null) {
  78. context.enterBlockingSection();
  79. final int res =
  80. s.getInputStream().read(buff.asBlock().getBytes(),
  81. ofs.asLong(),
  82. len.asLong());
  83. context.leaveBlockingSection();
  84. return Value.createFromLong(res);
  85. } else if (ds != null) {
  86. final DatagramPacket p =
  87. new DatagramPacket(buff.asBlock().getBytes(),
  88. ofs.asLong(),
  89. len.asLong());
  90. context.enterBlockingSection();
  91. ds.receive(p);
  92. context.leaveBlockingSection();
  93. return Value.createFromLong(p.getLength());
  94. } else {
  95. Unix.fail(ctxt, "recv", Unix.INVALID_DESCRIPTOR_MSG);
  96. return Value.UNIT; // never reached
  97. } // end if/elsif/else
  98. } catch (final InterruptedIOException iioe) {
  99. final FalseExit fe = FalseExit.createFromContext(context);
  100. fe.fillInStackTrace();
  101. throw fe;
  102. } catch (final IOException ioe) {
  103. context.leaveBlockingSection();
  104. Unix.fail(ctxt, "recv", ioe);
  105. return Value.UNIT; // never reached
  106. } // end try/catch
  107. } // end method 'unix_recv(CodeRunner, Value, Value, Value, ...)'
  108. /**
  109. * Sends a packet on a connected socket.
  110. * @param ctxt context
  111. * @param socket socket
  112. * @param buff buffer of data to send
  113. * @param ofs offset of data to send
  114. * @param len length of data to send
  115. * @param flags ignored
  116. * @return number of bytes sent
  117. * @throws Fail.Exception if an i/o error occurs
  118. */
  119. @Primitive
  120. public static Value unix_send(final CodeRunner ctxt,
  121. final Value socket,
  122. final Value buff,
  123. final Value ofs,
  124. final Value len,
  125. final Value flags)
  126. throws Fail.Exception, FalseExit {
  127. final Context context = ctxt.getContext();
  128. try {
  129. final Channel ch = context.getChannel(socket.asLong());
  130. if (ch == null) {
  131. Unix.fail(ctxt, "send", Unix.INVALID_DESCRIPTOR_MSG);
  132. return Value.UNIT; // never reached
  133. } // end if
  134. final java.net.Socket s = ch.asSocket();
  135. final DatagramSocket ds = ch.asDatagramSocket();
  136. if (s != null) {
  137. context.enterBlockingSection();
  138. s.getOutputStream().write(buff.asBlock().getBytes(),
  139. ofs.asLong(),
  140. len.asLong());
  141. context.leaveBlockingSection();
  142. return len;
  143. } else if (ds != null) {
  144. final DatagramPacket p =
  145. new DatagramPacket(buff.asBlock().getBytes(),
  146. ofs.asLong(),
  147. len.asLong(),
  148. ds.getRemoteSocketAddress());
  149. context.enterBlockingSection();
  150. ds.send(p);
  151. context.leaveBlockingSection();
  152. return len;
  153. } else {
  154. Unix.fail(ctxt, "send", Unix.INVALID_DESCRIPTOR_MSG);
  155. return Value.UNIT; // never reached
  156. } // end if/elsif/else
  157. } catch (final InterruptedIOException iioe) {
  158. final FalseExit fe = FalseExit.createFromContext(context);
  159. fe.fillInStackTrace();
  160. throw fe;
  161. } catch (final IOException ioe) {
  162. context.leaveBlockingSection();
  163. Unix.fail(ctxt, "send", ioe);
  164. return Value.UNIT; // never reached
  165. } // end try/catch
  166. } // end method 'unix_send(CodeRunner, Value, Value, Value, ...)'
  167. /**
  168. * Receives a packet on a datagram socket.
  169. * @param ctxt context
  170. * @param socket socket
  171. * @param buff buffer of data to receive
  172. * @param ofs offset of data to receive
  173. * @param len maximum length of data to receive
  174. * @param flags ignored
  175. * @return <i>(number_of_bytes_received, source_address)</i>
  176. * @throws Fail.Exception if an i/o error occurs
  177. */
  178. @Primitive
  179. public static Value unix_recvfrom(final CodeRunner ctxt,
  180. final Value socket,
  181. final Value buff,
  182. final Value ofs,
  183. final Value len,
  184. final Value flags)
  185. throws Fail.Exception, FalseExit {
  186. final Context context = ctxt.getContext();
  187. final Channel ch = context.getChannel(socket.asLong());
  188. if (ch == null) {
  189. Unix.fail(ctxt, "recvfrom", Unix.INVALID_DESCRIPTOR_MSG);
  190. return Value.UNIT; // never reached
  191. } // end if
  192. final DatagramSocket ds = ch.asDatagramSocket();
  193. if (ds == null) {
  194. Unix.fail(ctxt, "recvfrom", Unix.INVALID_DESCRIPTOR_MSG);
  195. return Value.UNIT; // never reached
  196. } // end if/elsif/else
  197. final DatagramPacket packet =
  198. new DatagramPacket(buff.asBlock().getBytes(),
  199. ofs.asLong(),
  200. len.asLong());
  201. try {
  202. context.enterBlockingSection();
  203. ds.receive(packet);
  204. context.leaveBlockingSection();
  205. final Block res = Block.createBlock(0,
  206. Value.createFromLong(packet.getLength()),
  207. Unix.createSockAddr(ctxt, (InetSocketAddress) packet.getSocketAddress()));
  208. return Value.createFromBlock(res);
  209. } catch (final InterruptedIOException iioe) {
  210. final FalseExit fe = FalseExit.createFromContext(context);
  211. fe.fillInStackTrace();
  212. throw fe;
  213. } catch (final IOException ioe) {
  214. context.leaveBlockingSection();
  215. Unix.fail(ctxt, "recvfrom", ioe);
  216. return Value.UNIT; // never reached
  217. } // end try/catch
  218. } // end method 'unix_recvfrom(CodeRunner, Value, Value, Value, ...)'
  219. /**
  220. * Sends a packet on a datagram socket.
  221. * @param ctxt context
  222. * @param socket socket
  223. * @param buff buffer of data to send
  224. * @param ofs offset of data to send
  225. * @param len length of data to send
  226. * @param flags ignored
  227. * @param dest destination socket
  228. * @return number of bytes sent
  229. * @throws Fail.Exception if an i/o error occurs
  230. */
  231. @Primitive
  232. public static Value unix_sendto(final CodeRunner ctxt,
  233. final Value socket,
  234. final Value buff,
  235. final Value ofs,
  236. final Value len,
  237. final Value flags,
  238. final Value dest)
  239. throws Fail.Exception, FalseExit {
  240. final Context context = ctxt.getContext();
  241. final Block destBlock = dest.asBlock();
  242. DatagramPacket packet;
  243. try {
  244. final SocketAddress addr =
  245. new InetSocketAddress(InetAddress.getByAddress(destBlock.get(0).asBlock().getBytes()),
  246. destBlock.get(1).asLong());
  247. packet = new DatagramPacket(buff.asBlock().getBytes(),
  248. ofs.asLong(),
  249. len.asLong(),
  250. addr);
  251. } catch (final IOException ioe) {
  252. Unix.fail(ctxt, "sendto", ioe);
  253. return Value.UNIT; // never reached
  254. } // end try/catch
  255. final Channel ch = context.getChannel(socket.asLong());
  256. if (ch == null) {
  257. Unix.fail(ctxt, "sendto", Unix.INVALID_DESCRIPTOR_MSG);
  258. return Value.UNIT; // never reached
  259. } // end if
  260. final DatagramSocket ds = ch.asDatagramSocket();
  261. if (ds == null) {
  262. Unix.fail(ctxt, "sendto", Unix.INVALID_DESCRIPTOR_MSG);
  263. return Value.UNIT; // never reached
  264. } // end if
  265. context.enterBlockingSection();
  266. try {
  267. ds.send(packet);
  268. context.leaveBlockingSection();
  269. return len;
  270. } catch (final InterruptedIOException iioe) {
  271. final FalseExit fe = FalseExit.createFromContext(context);
  272. fe.fillInStackTrace();
  273. throw fe;
  274. } catch (final IOException ioe) {
  275. context.leaveBlockingSection();
  276. Unix.fail(ctxt, "sendto", ioe);
  277. return Value.UNIT; // never reached
  278. } // end try/catch
  279. } // end method 'unix_sendto(CodeRunner, Value, Value, Value, ...)'
  280. /**
  281. * Exact synonym of {@link #unix_sendto(CodeRunner, Value, Value, Value, Value, Value, Value)}.
  282. */
  283. @Primitive
  284. public static Value unix_sendto_native(final CodeRunner ctxt,
  285. final Value socket,
  286. final Value buff,
  287. final Value ofs,
  288. final Value len,
  289. final Value flags,
  290. final Value dest)
  291. throws Fail.Exception, FalseExit {
  292. return unix_sendto(ctxt, socket, buff, ofs, len, flags, dest);
  293. } // end method 'unix_sendto_native(CodeRunner, Value, Value, Value, Value, ...)'
  294. } // end class 'Sendrecv'