* Copyright (c) 2015-2023 Google, Inc. All rights reserved.
* **********************************************************/
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* * Neither the name of Google, Inc. nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL VMWARE, INC. OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
* DAMAGE.
*/
#include <assert.h>
#include <map>
#include "ipc_reader.h"
#include "../common/memref.h"
#include "../common/utils.h"
namespace dynamorio {
namespace drmemtrace {
#ifdef VERBOSE
# include <iostream>
#endif
ipc_reader_t::ipc_reader_t()
: creation_success_(false)
{
}
ipc_reader_t::ipc_reader_t(const char *ipc_name, int verbosity)
: reader_t(verbosity, "IPC")
, pipe_(ipc_name)
{
creation_success_ = pipe_.create();
}
bool
ipc_reader_t::operator!()
{
return !creation_success_;
}
std::string
ipc_reader_t::get_stream_name() const
{
return pipe_.get_name();
}
bool
ipc_reader_t::init()
{
at_eof_ = false;
if (!creation_success_ || !pipe_.open_for_read())
return false;
pipe_.maximize_buffer();
cur_buf_ = buf_;
end_buf_ = buf_;
++*this;
return true;
}
ipc_reader_t::~ipc_reader_t()
{
pipe_.close();
pipe_.destroy();
}
trace_entry_t *
ipc_reader_t::read_next_entry()
{
trace_entry_t *from_queue = read_queued_entry();
if (from_queue != nullptr)
return from_queue;
++cur_buf_;
if (cur_buf_ >= end_buf_) {
ssize_t sz = pipe_.read(buf_, sizeof(buf_));
if (sz < 0 || sz % sizeof(*end_buf_) != 0) {
if (at_eof_)
return nullptr;
cur_buf_ = buf_;
cur_buf_->type = TRACE_TYPE_FOOTER;
cur_buf_->size = 0;
cur_buf_->addr = 0;
at_eof_ = true;
return cur_buf_;
}
cur_buf_ = buf_;
end_buf_ = buf_ + (sz / sizeof(*end_buf_));
}
if (cur_buf_->type == TRACE_TYPE_FOOTER)
at_eof_ = true;
return cur_buf_;
}
}
}