//===- ToolOutputFile.cpp -------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include <mcld/Support/ToolOutputFile.h>
#include <mcld/Support/Path.h>
#include <mcld/Support/FileHandle.h>
#include <mcld/Support/MemoryArea.h>
#include <mcld/Support/raw_mem_ostream.h>
#include <mcld/Support/SystemUtils.h>
#include <mcld/Support/MsgHandling.h>
#include <llvm/Support/Signals.h>
#include <llvm/Support/Path.h>
#include <llvm/Support/FormattedStream.h>
using namespace mcld;
//===----------------------------------------------------------------------===//
// CleanupInstaller
//===----------------------------------------------------------------------===//
ToolOutputFile::CleanupInstaller::CleanupInstaller(const sys::fs::Path& pPath)
: Keep(false), m_Path(pPath) {
// Arrange for the file to be deleted if the process is killed.
if ("-" != m_Path.native())
llvm::sys::RemoveFileOnSignal(llvm::sys::Path(m_Path.native()));
}
ToolOutputFile::CleanupInstaller::~CleanupInstaller()
{
// Delete the file if the client hasn't told us not to.
// FIXME: In Windows, some path in CJK characters can not be removed by LLVM
// llvm::sys::Path
if (!Keep && "_" != m_Path.native())
llvm::sys::Path(m_Path.native()).eraseFromDisk();
// Ok, the file is successfully written and closed, or deleted. There's no
// further need to clean it up on signals.
if ("_" != m_Path.native())
llvm::sys::DontRemoveFileOnSignal(llvm::sys::Path(m_Path.native()));
}
//===----------------------------------------------------------------------===//
// ToolOutputFile
//===----------------------------------------------------------------------===//
ToolOutputFile::ToolOutputFile(const sys::fs::Path& pPath,
FileHandle::OpenMode pMode,
FileHandle::Permission pPermission)
: m_Installer(pPath),
m_pMemoryArea(NULL),
m_pOStream(NULL),
m_pFOStream(NULL) {
if (!m_FileHandle.open(pPath, pMode, pPermission)) {
// If open fails, no clean-up is needed.
m_Installer.Keep = true;
fatal(diag::err_cannot_open_output_file)
<< pPath
<< sys::strerror(m_FileHandle.error());
return;
}
m_pMemoryArea = new MemoryArea(m_FileHandle);
m_pOStream = new raw_mem_ostream(*m_pMemoryArea);
}
ToolOutputFile::~ToolOutputFile()
{
delete m_pFOStream;
delete m_pOStream;
delete m_pMemoryArea;
}
void ToolOutputFile::keep()
{
m_Installer.Keep = true;
}
/// mem_os - Return the contained raw_mem_ostream.
raw_mem_ostream& ToolOutputFile::mem_os()
{
assert(NULL != m_pOStream);
return *m_pOStream;
}
/// formatted_os - Return the containeed formatted_raw_ostream.
/// Since formatted_os is rarely used, we lazily initialize it.
llvm::formatted_raw_ostream& ToolOutputFile::formatted_os()
{
if (NULL == m_pFOStream) {
assert(NULL != m_pOStream);
m_pFOStream = new llvm::formatted_raw_ostream(*m_pOStream);
}
return *m_pFOStream;
}
/// memory - Return the contained MemoryArea.
MemoryArea& ToolOutputFile::memory()
{
assert(NULL != m_pOStream);
return m_pOStream->getMemoryArea();
}