Index: native_client_sdk/src/libraries/nacl_io/mount_node_tcp.cc |
diff --git a/native_client_sdk/src/libraries/nacl_io/mount_node_tcp.cc b/native_client_sdk/src/libraries/nacl_io/mount_node_tcp.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..419f17aacf0e32da1e95eb9e320ca6e229c8fe1c |
--- /dev/null |
+++ b/native_client_sdk/src/libraries/nacl_io/mount_node_tcp.cc |
@@ -0,0 +1,142 @@ |
+// 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_tcp.h" |
+#include "nacl_io/pepper_interface.h" |
+ |
+namespace nacl_io { |
+ |
+MountNodeTCP::MountNodeTCP(Mount* mount) : MountNodeSocket(mount) {} |
+ |
+ |
+TCPSocketInterface* MountNodeTCP::TCPSocket() { |
+ if (mount_->ppapi() == NULL) |
+ return NULL; |
+ |
+ return mount_->ppapi()->GetTCPSocketInterface(); |
+} |
+ |
+Error MountNodeTCP::Init(int flags) { |
+ if (TCPSocket() == NULL) |
+ return EACCES; |
+ |
+ socket_resource_ = TCPSocket()->Create(mount_->ppapi()->GetInstance()); |
+ if (0 == socket_resource_) |
+ return EACCES; |
+ |
+ return 0; |
+} |
+ |
+Error MountNodeTCP::Bind(const struct sockaddr* addr, socklen_t len) { |
+ AUTO_LOCK(node_lock_); |
+ |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ /* Only bind once. */ |
+ if (local_addr_ != 0) |
+ return EINVAL; |
+ |
+ /* Lie, we won't known until we connect. */ |
binji
2013/08/09 19:28:06
s/known/know/
noelallen1
2013/08/09 22:53:22
Done.
|
+ return 0; |
+} |
+ |
+Error MountNodeTCP::Connect(const struct sockaddr* addr, socklen_t len) { |
+ AUTO_LOCK(node_lock_); |
+ |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ if (remote_addr_ != 0) |
+ return EISCONN; |
+ |
+ remote_addr_ = SockAddrToResource(addr, len); |
+ if (0 == remote_addr_) |
+ return EINVAL; |
+ |
+ int err = TCPSocket()->Connect(socket_resource_, |
+ remote_addr_, |
+ PP_BlockUntilComplete()); |
+ |
+ if (err == PP_OK) { |
+ local_addr_ = TCPSocket()->GetLocalAddress(socket_resource_); |
+ mount_->ppapi()->AddRefResource(local_addr_); |
binji
2013/08/09 19:28:06
did you mean to return early here? Otherwise this
noelallen1
2013/08/09 22:53:22
Done.
|
+ } |
+ |
+ mount_->ppapi()->ReleaseResource(remote_addr_); |
+ remote_addr_ = 0; |
+ return PPErrorToErrno(err); |
+} |
+ |
+Error MountNodeTCP::Recv(void* buf, size_t len, int flags, int* out_len) { |
+ AUTO_LOCK(node_lock_); |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ int err = TCPSocket()->Read(socket_resource_, |
binji
2013/08/09 19:28:06
nit: extra space after =
|
+ static_cast<char*>(buf), |
+ len & 0xFFFF, /* Max 64K at a time. */ |
binji
2013/08/09 19:28:06
this seems arbitrary, why?
noelallen1
2013/08/09 22:53:22
Max IP "packet" size is 64K.
Until we support str
binji
2013/08/09 23:08:27
OK, but you may want to use std::min then, this wi
|
+ PP_BlockUntilComplete()); |
+ if (err < 0) |
+ return PPErrorToErrno(err); |
+ |
+ *out_len = err; |
+ return 0; |
+} |
+ |
+Error MountNodeTCP::RecvFrom(void* buf, |
+ size_t len, |
+ int flags, |
+ struct sockaddr* src_addr, |
+ socklen_t* addrlen, |
+ int* out_len) { |
+ Error err = Recv(buf, len, flags, out_len); |
+ if (err == 0) |
+ GetPeerName(src_addr, addrlen); |
+ return err; |
+} |
+ |
+ |
+Error MountNodeTCP::Send(const void* buf, size_t len, int flags, int* out_len) { |
+ AUTO_LOCK(node_lock_); |
+ |
+ if (0 == socket_resource_) |
+ return EBADF; |
+ |
+ if (0 == remote_addr_) |
+ return ENOTCONN; |
+ |
+ int err = TCPSocket()->Write(socket_resource_, |
+ static_cast<const char*>(buf), |
+ len & 0xFFFF, /* Max 64K at a time. */ |
+ PP_BlockUntilComplete()); |
+ if (err >= 0) { |
binji
2013/08/09 19:28:06
Invert this to match similar check in Recv() above
noelallen1
2013/08/09 22:53:22
Done.
|
+ *out_len = err; |
+ return 0; |
+ } |
+ |
+ return PPErrorToErrno(err); |
+} |
+ |
+Error MountNodeTCP::SendTo(const void* buf, |
+ size_t len, |
+ int flags, |
+ const struct sockaddr* dest_addr, |
+ socklen_t addrlen, |
+ int* out_len) { |
+ return Send(buf, len, flags, out_len); |
+} |
+ |
+} // namespace nacl_io |
+ |
+#endif // PROVIDES_SOCKET_API |