Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(32)

Unified Diff: native_client_sdk/src/libraries/nacl_io/mount_node_udp.cc

Issue 22587003: [NaCl SDK] Add UDP and TCP Sockets (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Merge to browser tester fix. Created 7 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
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..310a2bdf06581cbfb87039872211109e1af1246d
--- /dev/null
+++ b/native_client_sdk/src/libraries/nacl_io/mount_node_udp.cc
@@ -0,0 +1,188 @@
+// 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 <algorithm>
+
+#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());
+ if (err != 0) {
+ mount_->ppapi()->ReleaseResource(out_addr);
+ return PPErrorToErrno(err);
+ }
+
+ local_addr_ = out_addr;
+ return 0;
+}
+
+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 capped_len = static_cast<int32_t>(std::min(len, MAX_SOCK_TRANSFER));
+ int err = UDPSocket()->RecvFrom(socket_resource_,
+ static_cast<char*>(buf),
+ capped_len,
+ 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 local_len = 0;
+ PP_Resource addr = 0;
+
+ int err = RecvFromHelper(buf, len, flags, &addr, &local_len);
+ if (err < 0)
+ return PPErrorToErrno(err);
+
+ /* If "connected" then only receive packets from the given remote. */
+ bool same = IsEquivalentAddress(addr, remote_addr_);
+ mount_->ppapi()->ReleaseResource(addr);
+
+ if (remote_addr_ != 0 && same)
+ continue;
+
+ *out_len = local_len;
+ 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);
+
+ mount_->ppapi()->ReleaseResource(addr);
+ 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 capped_len = static_cast<int32_t>(std::min(len, MAX_SOCK_TRANSFER));
+ int err = UDPSocket()->SendTo(socket_resource_,
+ static_cast<const char*>(buf),
+ capped_len,
+ addr,
+ PP_BlockUntilComplete());
+ if (err < 0)
+ return PPErrorToErrno(err);
+
+ *out_len = err;
+ return 0;
+}
+
+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);
+ 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
« no previous file with comments | « native_client_sdk/src/libraries/nacl_io/mount_node_udp.h ('k') | native_client_sdk/src/libraries/nacl_io/mount_socket.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698