|  | //===-- SBStream.cpp ----------------------------------------*- C++ -*-===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "lldb/API/SBStream.h" | 
|  |  | 
|  | #include "lldb/Core/Error.h" | 
|  | #include "lldb/Core/Stream.h" | 
|  | #include "lldb/Core/StreamFile.h" | 
|  | #include "lldb/Core/StreamString.h" | 
|  |  | 
|  | using namespace lldb; | 
|  | using namespace lldb_private; | 
|  |  | 
|  | SBStream::SBStream() : m_opaque_ap(new StreamString()), m_is_file(false) {} | 
|  |  | 
|  | SBStream::SBStream(SBStream &&rhs) | 
|  | : m_opaque_ap(std::move(rhs.m_opaque_ap)), m_is_file(rhs.m_is_file) {} | 
|  |  | 
|  | SBStream::~SBStream() {} | 
|  |  | 
|  | bool SBStream::IsValid() const { return (m_opaque_ap.get() != NULL); } | 
|  |  | 
|  | // If this stream is not redirected to a file, it will maintain a local | 
|  | // cache for the stream data which can be accessed using this accessor. | 
|  | const char *SBStream::GetData() { | 
|  | if (m_is_file || m_opaque_ap.get() == NULL) | 
|  | return NULL; | 
|  |  | 
|  | return static_cast<StreamString *>(m_opaque_ap.get())->GetData(); | 
|  | } | 
|  |  | 
|  | // If this stream is not redirected to a file, it will maintain a local | 
|  | // cache for the stream output whose length can be accessed using this | 
|  | // accessor. | 
|  | size_t SBStream::GetSize() { | 
|  | if (m_is_file || m_opaque_ap.get() == NULL) | 
|  | return 0; | 
|  |  | 
|  | return static_cast<StreamString *>(m_opaque_ap.get())->GetSize(); | 
|  | } | 
|  |  | 
|  | void SBStream::Printf(const char *format, ...) { | 
|  | if (!format) | 
|  | return; | 
|  | va_list args; | 
|  | va_start(args, format); | 
|  | ref().PrintfVarArg(format, args); | 
|  | va_end(args); | 
|  | } | 
|  |  | 
|  | void SBStream::RedirectToFile(const char *path, bool append) { | 
|  | if (path == nullptr) | 
|  | return; | 
|  |  | 
|  | std::string local_data; | 
|  | if (m_opaque_ap.get()) { | 
|  | // See if we have any locally backed data. If so, copy it so we can then | 
|  | // redirect it to the file so we don't lose the data | 
|  | if (!m_is_file) | 
|  | local_data.swap( | 
|  | static_cast<StreamString *>(m_opaque_ap.get())->GetString()); | 
|  | } | 
|  | StreamFile *stream_file = new StreamFile; | 
|  | uint32_t open_options = File::eOpenOptionWrite | File::eOpenOptionCanCreate; | 
|  | if (append) | 
|  | open_options |= File::eOpenOptionAppend; | 
|  | else | 
|  | open_options |= File::eOpenOptionTruncate; | 
|  | stream_file->GetFile().Open(path, open_options, | 
|  | lldb::eFilePermissionsFileDefault); | 
|  |  | 
|  | m_opaque_ap.reset(stream_file); | 
|  |  | 
|  | if (m_opaque_ap.get()) { | 
|  | m_is_file = true; | 
|  |  | 
|  | // If we had any data locally in our StreamString, then pass that along to | 
|  | // the to new file we are redirecting to. | 
|  | if (!local_data.empty()) | 
|  | m_opaque_ap->Write(&local_data[0], local_data.size()); | 
|  | } else | 
|  | m_is_file = false; | 
|  | } | 
|  |  | 
|  | void SBStream::RedirectToFileHandle(FILE *fh, bool transfer_fh_ownership) { | 
|  | if (fh == nullptr) | 
|  | return; | 
|  |  | 
|  | std::string local_data; | 
|  | if (m_opaque_ap.get()) { | 
|  | // See if we have any locally backed data. If so, copy it so we can then | 
|  | // redirect it to the file so we don't lose the data | 
|  | if (!m_is_file) | 
|  | local_data.swap( | 
|  | static_cast<StreamString *>(m_opaque_ap.get())->GetString()); | 
|  | } | 
|  | m_opaque_ap.reset(new StreamFile(fh, transfer_fh_ownership)); | 
|  |  | 
|  | if (m_opaque_ap.get()) { | 
|  | m_is_file = true; | 
|  |  | 
|  | // If we had any data locally in our StreamString, then pass that along to | 
|  | // the to new file we are redirecting to. | 
|  | if (!local_data.empty()) | 
|  | m_opaque_ap->Write(&local_data[0], local_data.size()); | 
|  | } else | 
|  | m_is_file = false; | 
|  | } | 
|  |  | 
|  | void SBStream::RedirectToFileDescriptor(int fd, bool transfer_fh_ownership) { | 
|  | std::string local_data; | 
|  | if (m_opaque_ap.get()) { | 
|  | // See if we have any locally backed data. If so, copy it so we can then | 
|  | // redirect it to the file so we don't lose the data | 
|  | if (!m_is_file) | 
|  | local_data.swap( | 
|  | static_cast<StreamString *>(m_opaque_ap.get())->GetString()); | 
|  | } | 
|  |  | 
|  | m_opaque_ap.reset(new StreamFile(::fdopen(fd, "w"), transfer_fh_ownership)); | 
|  | if (m_opaque_ap.get()) { | 
|  | m_is_file = true; | 
|  |  | 
|  | // If we had any data locally in our StreamString, then pass that along to | 
|  | // the to new file we are redirecting to. | 
|  | if (!local_data.empty()) | 
|  | m_opaque_ap->Write(&local_data[0], local_data.size()); | 
|  | } else | 
|  | m_is_file = false; | 
|  | } | 
|  |  | 
|  | lldb_private::Stream *SBStream::operator->() { return m_opaque_ap.get(); } | 
|  |  | 
|  | lldb_private::Stream *SBStream::get() { return m_opaque_ap.get(); } | 
|  |  | 
|  | lldb_private::Stream &SBStream::ref() { | 
|  | if (m_opaque_ap.get() == NULL) | 
|  | m_opaque_ap.reset(new StreamString()); | 
|  | return *m_opaque_ap.get(); | 
|  | } | 
|  |  | 
|  | void SBStream::Clear() { | 
|  | if (m_opaque_ap.get()) { | 
|  | // See if we have any locally backed data. If so, copy it so we can then | 
|  | // redirect it to the file so we don't lose the data | 
|  | if (m_is_file) | 
|  | m_opaque_ap.reset(); | 
|  | else | 
|  | static_cast<StreamString *>(m_opaque_ap.get())->GetString().clear(); | 
|  | } | 
|  | } |