blob: e63cccd83ef35f440e13ea1f1a51b906bfbacce3 [file] [log] [blame]
// Copyright 2020 The Fuchsia 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 "src/lib/storage/vfs/cpp/stream_file_connection.h"
#include <fcntl.h>
#include <fidl/fuchsia.io/cpp/wire.h>
#include <lib/fdio/io.h>
#include <lib/fdio/vfs.h>
#include <lib/zx/handle.h>
#include <lib/zx/result.h>
#include <limits.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <zircon/assert.h>
#include <zircon/rights.h>
#include <memory>
#include <type_traits>
#include <utility>
#include <fbl/string_buffer.h>
#include "src/lib/storage/vfs/cpp/connection.h"
#include "src/lib/storage/vfs/cpp/debug.h"
#include "src/lib/storage/vfs/cpp/file_connection.h"
#include "src/lib/storage/vfs/cpp/vfs_types.h"
#include "src/lib/storage/vfs/cpp/vnode.h"
namespace fio = fuchsia_io;
namespace fs {
namespace internal {
StreamFileConnection::StreamFileConnection(fs::FuchsiaVfs* vfs, fbl::RefPtr<fs::Vnode> vnode,
zx::stream stream, VnodeProtocol protocol,
VnodeConnectionOptions options, zx_koid_t koid)
: FileConnection(vfs, std::move(vnode), protocol, options, koid), stream_(std::move(stream)) {}
zx_status_t StreamFileConnection::ReadInternal(void* data, size_t len, size_t* out_actual) {
FS_PRETTY_TRACE_DEBUG("[FileRead] options: ", options());
if (options().flags.node_reference) {
return ZX_ERR_BAD_HANDLE;
}
if (!options().rights.read) {
return ZX_ERR_BAD_HANDLE;
}
if (len > fio::wire::kMaxBuf) {
return ZX_ERR_INVALID_ARGS;
}
zx_iovec_t vector = {
.buffer = data,
.capacity = len,
};
zx_status_t status = stream_.readv(0, &vector, 1, out_actual);
if (status == ZX_OK) {
ZX_DEBUG_ASSERT(*out_actual <= len);
}
return status;
}
void StreamFileConnection::Read(ReadRequestView request, ReadCompleter::Sync& completer) {
uint8_t data[fio::wire::kMaxBuf];
size_t actual = 0;
zx_status_t status = ReadInternal(data, request->count, &actual);
if (status != ZX_OK) {
completer.ReplyError(status);
} else {
completer.ReplySuccess(fidl::VectorView<uint8_t>::FromExternal(data, actual));
}
}
zx_status_t StreamFileConnection::ReadAtInternal(void* data, size_t len, size_t offset,
size_t* out_actual) {
FS_PRETTY_TRACE_DEBUG("[FileReadAt] options: ", options());
if (options().flags.node_reference) {
return ZX_ERR_BAD_HANDLE;
}
if (!options().rights.read) {
return ZX_ERR_BAD_HANDLE;
}
if (len > fio::wire::kMaxBuf) {
return ZX_ERR_INVALID_ARGS;
}
zx_iovec_t vector = {
.buffer = data,
.capacity = len,
};
zx_status_t status = stream_.readv_at(0, offset, &vector, 1, out_actual);
if (status == ZX_OK) {
ZX_DEBUG_ASSERT(*out_actual <= len);
}
return status;
}
void StreamFileConnection::ReadAt(ReadAtRequestView request, ReadAtCompleter::Sync& completer) {
uint8_t data[fio::wire::kMaxBuf];
size_t actual = 0;
zx_status_t status = ReadAtInternal(data, request->count, request->offset, &actual);
if (status != ZX_OK) {
completer.ReplyError(status);
} else {
completer.ReplySuccess(fidl::VectorView<uint8_t>::FromExternal(data, actual));
}
}
zx_status_t StreamFileConnection::WriteInternal(const void* data, size_t len, size_t* out_actual) {
FS_PRETTY_TRACE_DEBUG("[FileWrite] options: ", options());
if (options().flags.node_reference) {
return ZX_ERR_BAD_HANDLE;
}
if (!options().rights.write) {
return ZX_ERR_BAD_HANDLE;
}
zx_iovec_t vector = {
.buffer = const_cast<void*>(data),
.capacity = len,
};
zx_status_t status = stream_.writev(0, &vector, 1, out_actual);
if (status == ZX_OK) {
ZX_DEBUG_ASSERT(*out_actual <= len);
vnode()->DidModifyStream();
}
return status;
}
void StreamFileConnection::Write(WriteRequestView request, WriteCompleter::Sync& completer) {
size_t actual = 0u;
zx_status_t status = WriteInternal(request->data.data(), request->data.count(), &actual);
if (status != ZX_OK) {
completer.ReplyError(status);
} else {
completer.ReplySuccess(actual);
}
}
zx_status_t StreamFileConnection::WriteAtInternal(const void* data, size_t len, size_t offset,
size_t* out_actual) {
FS_PRETTY_TRACE_DEBUG("[FileWriteAt] options: ", options());
if (options().flags.node_reference) {
return ZX_ERR_BAD_HANDLE;
}
if (!options().rights.write) {
return ZX_ERR_BAD_HANDLE;
}
zx_iovec_t vector = {
.buffer = const_cast<void*>(data),
.capacity = len,
};
zx_status_t status = stream_.writev_at(0, offset, &vector, 1, out_actual);
if (status == ZX_OK) {
ZX_DEBUG_ASSERT(*out_actual <= len);
vnode()->DidModifyStream();
}
return status;
}
void StreamFileConnection::WriteAt(WriteAtRequestView request, WriteAtCompleter::Sync& completer) {
size_t actual = 0;
zx_status_t status =
WriteAtInternal(request->data.data(), request->data.count(), request->offset, &actual);
if (status != ZX_OK) {
completer.ReplyError(status);
} else {
completer.ReplySuccess(actual);
}
}
void StreamFileConnection::Seek(SeekRequestView request, SeekCompleter::Sync& completer) {
FS_PRETTY_TRACE_DEBUG("[FileSeek] options: ", options());
if (options().flags.node_reference) {
completer.ReplyError(ZX_ERR_BAD_HANDLE);
return;
}
zx_off_t seek = 0u;
zx_status_t status =
stream_.seek(static_cast<zx_stream_seek_origin_t>(request->origin), request->offset, &seek);
if (status != ZX_OK) {
completer.ReplyError(status);
} else {
completer.ReplySuccess(seek);
}
}
void StreamFileConnection::GetFlags(GetFlagsCompleter::Sync& completer) {
zx::result result = GetFlagsInternal();
if (result.is_error()) {
completer.Reply(result.status_value(), {});
} else {
completer.Reply(ZX_OK, result.value());
}
}
void StreamFileConnection::SetFlags(SetFlagsRequestView request,
SetFlagsCompleter::Sync& completer) {
completer.Reply(SetFlagsInternal(request->flags).status_value());
}
zx::result<fuchsia_io::wire::OpenFlags> StreamFileConnection::GetFlagsInternal() {
zx::result flags = NodeGetFlags();
if constexpr (ZX_DEBUG_ASSERT_IMPLEMENTED) {
// Validate that the connection's append mode and the stream's append mode match.
if (flags.is_ok()) {
uint8_t mode_append;
if (zx_status_t status = stream_.get_prop_mode_append(&mode_append); status != ZX_OK) {
return zx::error(status);
}
bool stream_append = mode_append;
bool flags_append = static_cast<bool>(flags.value() & fuchsia_io::wire::OpenFlags::kAppend);
ZX_ASSERT_MSG(stream_append == flags_append, "stream append: %d flags append: %d",
stream_append, flags_append);
}
}
return flags;
}
zx::result<> StreamFileConnection::SetFlagsInternal(fuchsia_io::wire::OpenFlags flags) {
auto new_options = VnodeConnectionOptions::FromIoV1Flags(flags);
bool append = new_options.flags.append;
auto status = zx::make_result(stream_.set_prop_mode_append(append));
if (status.is_ok()) {
set_append(append);
}
return status;
}
zx::result<VnodeRepresentation> StreamFileConnection::NodeDescribe() {
zx::result representation = Connection::NodeDescribe();
if (representation.is_error()) {
return representation.take_error();
}
if (representation->is_file()) {
ZX_DEBUG_ASSERT_MSG(!representation->file().stream.is_valid(),
"The stream should only be set by the connection");
if (vnode()->SupportsClientSideStreams()) {
if (zx_status_t status =
stream_.duplicate(ZX_RIGHT_SAME_RIGHTS, &representation->file().stream);
status != ZX_OK) {
return zx::error(status);
}
}
}
return representation;
}
} // namespace internal
} // namespace fs