blob: f6669ec989c64f230b76801a20b50fbe2429a15b [file] [log] [blame]
// Copyright 2017 The Fuchsia Authors
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#include <err.h>
#include <inttypes.h>
#include <platform.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <trace.h>
#include <lib/user_copy.h>
#include <lib/user_copy/user_ptr.h>
#include <magenta/handle_owner.h>
#include <magenta/process_dispatcher.h>
#include <magenta/socket_dispatcher.h>
#include <magenta/syscalls/policy.h>
#include <mxtl/ref_ptr.h>
#include "syscalls_priv.h"
#define LOCAL_TRACE 0
mx_status_t sys_socket_create(uint32_t options, user_ptr<mx_handle_t> _out0, user_ptr<mx_handle_t> _out1) {
LTRACEF("entry out_handles %p, %p\n", _out0.get(), _out1.get());
auto up = ProcessDispatcher::GetCurrent();
mx_status_t res = up->QueryPolicy(MX_POL_NEW_SOCKET);
if (res != MX_OK)
return res;
mxtl::RefPtr<Dispatcher> socket0, socket1;
mx_rights_t rights;
mx_status_t result = SocketDispatcher::Create(options, &socket0, &socket1, &rights);
if (result != MX_OK)
return result;
HandleOwner h0(MakeHandle(mxtl::move(socket0), rights));
if (!h0)
return MX_ERR_NO_MEMORY;
HandleOwner h1(MakeHandle(mxtl::move(socket1), rights));
if (!h1)
return MX_ERR_NO_MEMORY;
if (_out0.copy_to_user(up->MapHandleToValue(h0)) != MX_OK)
return MX_ERR_INVALID_ARGS;
if (_out1.copy_to_user(up->MapHandleToValue(h1)) != MX_OK)
return MX_ERR_INVALID_ARGS;
up->AddHandle(mxtl::move(h0));
up->AddHandle(mxtl::move(h1));
return MX_OK;
}
mx_status_t sys_socket_write(mx_handle_t handle, uint32_t options,
user_ptr<const void> _buffer, size_t size,
user_ptr<size_t> _actual) {
LTRACEF("handle %x\n", handle);
if ((size > 0u) && !_buffer)
return MX_ERR_INVALID_ARGS;
auto up = ProcessDispatcher::GetCurrent();
mxtl::RefPtr<SocketDispatcher> socket;
mx_status_t status = up->GetDispatcherWithRights(handle, MX_RIGHT_WRITE, &socket);
if (status != MX_OK)
return status;
size_t nwritten;
switch (options) {
case 0:
status = socket->Write(_buffer, size, &nwritten);
break;
case MX_SOCKET_CONTROL:
status = socket->WriteControl(_buffer, size);
if (status == MX_OK)
nwritten = size;
break;
case MX_SOCKET_SHUTDOWN_WRITE:
case MX_SOCKET_SHUTDOWN_READ:
case MX_SOCKET_SHUTDOWN_READ | MX_SOCKET_SHUTDOWN_WRITE:
if (size == 0)
return socket->Shutdown(options & MX_SOCKET_SHUTDOWN_MASK);
// fallthrough
default:
return MX_ERR_INVALID_ARGS;
}
// Caller may ignore results if desired.
if (status == MX_OK && _actual)
status = _actual.copy_to_user(nwritten);
return status;
}
mx_status_t sys_socket_read(mx_handle_t handle, uint32_t options,
user_ptr<void> _buffer, size_t size,
user_ptr<size_t> _actual) {
LTRACEF("handle %x\n", handle);
if (options)
if (!_buffer && size > 0)
return MX_ERR_INVALID_ARGS;
auto up = ProcessDispatcher::GetCurrent();
mxtl::RefPtr<SocketDispatcher> socket;
mx_status_t status = up->GetDispatcherWithRights(handle, MX_RIGHT_READ, &socket);
if (status != MX_OK)
return status;
size_t nread;
switch (options) {
case 0:
status = socket->Read(_buffer, size, &nread);
break;
case MX_SOCKET_CONTROL:
status = socket->ReadControl(_buffer, size, &nread);
break;
default:
return MX_ERR_INVALID_ARGS;
}
// Caller may ignore results if desired.
if (status == MX_OK && _actual)
status = _actual.copy_to_user(nread);
return status;
}