Index: native_client_sdk/src/libraries/nacl_io/mount_node_udp.cc |
diff --git a/native_client_sdk/src/libraries/nacl_io/mount_node_udp.cc b/native_client_sdk/src/libraries/nacl_io/mount_node_udp.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..c4ba0222a8228987cdd4a1d6d9da2902996efd8d |
--- /dev/null |
+++ b/native_client_sdk/src/libraries/nacl_io/mount_node_udp.cc |
@@ -0,0 +1,176 @@ |
+// Copyright (c) 2013 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+ |
+#include "nacl_io/ossocket.h" |
+#ifdef PROVIDES_SOCKET_API |
+ |
+#include <errno.h> |
+#include <string.h> |
+ |
+#include "nacl_io/mount.h" |
+#include "nacl_io/mount_node_socket.h" |
+#include "nacl_io/mount_node_udp.h" |
+#include "nacl_io/pepper_interface.h" |
+ |
+namespace nacl_io { |
+ |
+MountNodeUDP::MountNodeUDP(Mount* mount) : MountNodeSocket(mount) {} |
+ |
+ |
+UDPSocketInterface* MountNodeUDP::UDPSocket() { |
+ if (mount_->ppapi() == NULL) |
+ return NULL; |
+ |
+ return mount_->ppapi()->GetUDPSocketInterface(); |
+} |
+ |
+Error MountNodeUDP::Init(int flags) { |
+ if (UDPSocket() == NULL) |
+ return EACCES; |
+ |
+ socket_resource_ = UDPSocket()->Create(mount_->ppapi()->GetInstance()); |
+ if (0 == socket_resource_) |
+ return EACCES; |
+ |
+ return 0; |
+} |
+ |
+Error MountNodeUDP::Bind(const struct sockaddr* addr, socklen_t len) { |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ /* Only bind once. */ |
+ if (local_addr_ != 0) |
+ return EINVAL; |
+ |
+ PP_Resource out_addr = SockAddrToResource(addr, len); |
+ if (0 == out_addr) |
+ return EINVAL; |
+ |
+ int err = UDPSocket()->Bind(socket_resource_, |
+ out_addr, |
+ PP_BlockUntilComplete()); |
+ return PPErrorToErrno(err); |
+} |
+ |
+Error MountNodeUDP::Connect(const struct sockaddr* addr, socklen_t len) { |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ /* Connect for UDP is the default dest, it's legal to change it. */ |
+ if (remote_addr_ != 0) { |
+ mount_->ppapi()->ReleaseResource(remote_addr_); |
+ remote_addr_ = 0; |
+ } |
+ |
+ remote_addr_ = SockAddrToResource(addr, len); |
+ if (0 == remote_addr_) |
+ return EINVAL; |
+ |
+ return 0; |
+} |
+ |
+Error MountNodeUDP::RecvFromHelper(void* buf, |
+ size_t len, |
+ int flags, |
+ PP_Resource* out_addr, |
+ int* out_len) { |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ int err = UDPSocket()->RecvFrom(socket_resource_, |
+ static_cast<char*>(buf), |
+ len & 0xFFFF, |
+ out_addr, |
+ PP_BlockUntilComplete()); |
+ if (err < 0) |
+ return PPErrorToErrno(err); |
+ |
+ *out_len = err; |
+ return 0; |
+} |
+ |
+Error MountNodeUDP::Recv(void* buf, size_t len, int flags, int* out_len) { |
+ while (1) { |
+ int out_len = 0; |
binji
2013/08/09 19:28:06
this shadows out_len above
noelallen1
2013/08/09 22:53:22
Done.
|
+ PP_Resource addr = 0; |
+ |
+ int err = RecvFromHelper(buf, len, flags, &addr, &out_len); |
+ if (err < 0) |
+ return PPErrorToErrno(err); |
+ |
+ /* If "connected" then only receive packets from the given remote. */ |
+ if (remote_addr_ != 0 && !EquivalentAddress(addr, remote_addr_)) |
+ continue; |
+ |
+ return 0; |
+ } |
+} |
+ |
+Error MountNodeUDP::RecvFrom(void* buf, |
+ size_t len, |
+ int flags, |
+ struct sockaddr* src_addr, |
+ socklen_t* addrlen, |
+ int* out_len) { |
+ PP_Resource addr = 0; |
+ int err = RecvFromHelper(buf, len, flags, &addr, out_len); |
+ if (err < 0) |
+ return PPErrorToErrno(err); |
+ |
+ if (src_addr) |
+ *addrlen = ResourceToSockAddr(addr, *addrlen, src_addr); |
+ |
binji
2013/08/09 19:28:06
need to release resource?
noelallen1
2013/08/09 22:53:22
Done.
|
+ return 0; |
+} |
+ |
+ |
+Error MountNodeUDP::SendToHelper(const void* buf, |
+ size_t len, |
+ int flags, |
+ PP_Resource addr, |
+ int* out_len) { |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ if (0 == addr) |
+ return ENOTCONN; |
+ |
+ int err = mount_->ppapi()->GetUDPSocketInterface()->SendTo( |
+ socket_resource_, |
binji
2013/08/09 19:28:06
nit: weird indent
noelallen1
2013/08/09 22:53:22
Done.
|
+ static_cast<const char*>(buf), |
+ len & 0xFFFF, |
+ addr, |
+ PP_BlockUntilComplete()); |
+ if (err >= 0) { |
binji
2013/08/09 19:28:06
invert check to match Recv
noelallen1
2013/08/09 22:53:22
Done.
|
+ *out_len = err; |
+ return 0; |
+ } |
+ |
+ return PPErrorToErrno(err); |
+} |
+ |
+Error MountNodeUDP::Send(const void* buf, size_t len, int flags, int* out_len) { |
+ return SendToHelper(buf, len, flags, remote_addr_, out_len); |
+} |
+ |
+Error MountNodeUDP::SendTo(const void* buf, |
+ size_t len, |
+ int flags, |
+ const struct sockaddr* dest_addr, |
+ socklen_t addrlen, |
+ int* out_len) { |
+ PP_Resource out_addr = SockAddrToResource(dest_addr, addrlen); |
binji
2013/08/09 19:28:06
nit: extra space after =
binji
2013/08/09 19:28:06
use ScopedResource instead?
noelallen1
2013/08/09 22:53:22
Done.
|
+ if (0 == out_addr) |
+ return EINVAL; |
+ |
+ Error err = SendToHelper(buf, len, flags, out_addr, out_len); |
+ mount_->ppapi()->ReleaseResource(out_addr); |
+ return err; |
+} |
+ |
+} // namespace nacl_io |
+ |
+#endif // PROVIDES_SOCKET_API |