mirror of
https://github.com/intel/llvm.git
synced 2026-01-13 11:02:04 +08:00
If we open a `NativeFile` with a `FILE*`, the OpenOptions default to
`eOpenOptionReadOnly`. This is an issue in python scripts if you try to
write to one of the files like `print("Hi",
file=lldb.debugger.GetOutputFileHandle())`.
To address this, we need to specify the access mode whenever we create a
`NativeFile` from a `FILE*`. I also added an assert on the `NativeFile`
that validates the file is opened with the correct access mode and
updated `NativeFile::Read` and `NativeFile::Write` to check the access
mode.
Before these changes:
```
$ lldb -b -O 'script lldb.debugger.GetOutputFileHandle().write("abc")'
(lldb) script lldb.debugger.GetOutputFileHandle().write("abc")
Traceback (most recent call last):
File "<input>", line 1, in <module>
io.UnsupportedOperation: not writable
```
After:
```
$ lldb -b -O 'script lldb.debugger.GetOutputFileHandle().write("abc")'
(lldb) script lldb.debugger.GetOutputFileHandle().write("abc")
abc3
```
Fixes #122387
140 lines
3.6 KiB
C++
140 lines
3.6 KiB
C++
//===-- SBFile.cpp --------------------------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "lldb/API/SBFile.h"
|
|
#include "lldb/API/SBError.h"
|
|
#include "lldb/Host/File.h"
|
|
#include "lldb/Utility/Instrumentation.h"
|
|
|
|
using namespace lldb;
|
|
using namespace lldb_private;
|
|
|
|
SBFile::~SBFile() = default;
|
|
|
|
SBFile::SBFile(FileSP file_sp) : m_opaque_sp(file_sp) {
|
|
// We have no way to capture the incoming FileSP as the class isn't
|
|
// instrumented, so pretend that it's always null.
|
|
LLDB_INSTRUMENT_VA(this, file_sp);
|
|
}
|
|
|
|
SBFile::SBFile(const SBFile &rhs) : m_opaque_sp(rhs.m_opaque_sp) {
|
|
LLDB_INSTRUMENT_VA(this, rhs);
|
|
}
|
|
|
|
SBFile &SBFile ::operator=(const SBFile &rhs) {
|
|
LLDB_INSTRUMENT_VA(this, rhs);
|
|
|
|
if (this != &rhs)
|
|
m_opaque_sp = rhs.m_opaque_sp;
|
|
return *this;
|
|
}
|
|
|
|
SBFile::SBFile() { LLDB_INSTRUMENT_VA(this); }
|
|
|
|
SBFile::SBFile(FILE *file, bool transfer_ownership) {
|
|
LLDB_INSTRUMENT_VA(this, file, transfer_ownership);
|
|
|
|
// For backwards comptability, this defaulted to ReadOnly previously.
|
|
m_opaque_sp = std::make_shared<NativeFile>(file, File::eOpenOptionReadOnly,
|
|
transfer_ownership);
|
|
}
|
|
|
|
SBFile::SBFile(FILE *file, const char *mode, bool transfer_ownership) {
|
|
LLDB_INSTRUMENT_VA(this, file, transfer_ownership);
|
|
|
|
auto options = File::GetOptionsFromMode(mode);
|
|
if (!options) {
|
|
llvm::consumeError(options.takeError());
|
|
return;
|
|
}
|
|
|
|
m_opaque_sp =
|
|
std::make_shared<NativeFile>(file, options.get(), transfer_ownership);
|
|
}
|
|
|
|
SBFile::SBFile(int fd, const char *mode, bool transfer_owndership) {
|
|
LLDB_INSTRUMENT_VA(this, fd, mode, transfer_owndership);
|
|
|
|
auto options = File::GetOptionsFromMode(mode);
|
|
if (!options) {
|
|
llvm::consumeError(options.takeError());
|
|
return;
|
|
}
|
|
m_opaque_sp =
|
|
std::make_shared<NativeFile>(fd, options.get(), transfer_owndership);
|
|
}
|
|
|
|
SBError SBFile::Read(uint8_t *buf, size_t num_bytes, size_t *bytes_read) {
|
|
LLDB_INSTRUMENT_VA(this, buf, num_bytes, bytes_read);
|
|
|
|
SBError error;
|
|
if (!m_opaque_sp) {
|
|
error = Status::FromErrorString("invalid SBFile");
|
|
*bytes_read = 0;
|
|
} else {
|
|
error.SetError(m_opaque_sp->Read(buf, num_bytes));
|
|
*bytes_read = num_bytes;
|
|
}
|
|
return error;
|
|
}
|
|
|
|
SBError SBFile::Write(const uint8_t *buf, size_t num_bytes,
|
|
size_t *bytes_written) {
|
|
LLDB_INSTRUMENT_VA(this, buf, num_bytes, bytes_written);
|
|
|
|
SBError error;
|
|
if (!m_opaque_sp) {
|
|
error = Status::FromErrorString("invalid SBFile");
|
|
*bytes_written = 0;
|
|
} else {
|
|
error.SetError(m_opaque_sp->Write(buf, num_bytes));
|
|
*bytes_written = num_bytes;
|
|
}
|
|
return error;
|
|
}
|
|
|
|
SBError SBFile::Flush() {
|
|
LLDB_INSTRUMENT_VA(this);
|
|
|
|
SBError error;
|
|
if (!m_opaque_sp) {
|
|
error = Status::FromErrorString("invalid SBFile");
|
|
} else {
|
|
error.SetError(m_opaque_sp->Flush());
|
|
}
|
|
return error;
|
|
}
|
|
|
|
bool SBFile::IsValid() const {
|
|
LLDB_INSTRUMENT_VA(this);
|
|
return m_opaque_sp && m_opaque_sp->IsValid();
|
|
}
|
|
|
|
SBError SBFile::Close() {
|
|
LLDB_INSTRUMENT_VA(this);
|
|
SBError error;
|
|
if (m_opaque_sp)
|
|
error.SetError(m_opaque_sp->Close());
|
|
return error;
|
|
}
|
|
|
|
SBFile::operator bool() const {
|
|
LLDB_INSTRUMENT_VA(this);
|
|
return IsValid();
|
|
}
|
|
|
|
bool SBFile::operator!() const {
|
|
LLDB_INSTRUMENT_VA(this);
|
|
return !IsValid();
|
|
}
|
|
|
|
FileSP SBFile::GetFile() const {
|
|
LLDB_INSTRUMENT_VA(this);
|
|
return m_opaque_sp;
|
|
}
|