Chromium Code Reviews| 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 |