Commit 1081df6e authored by tzik's avatar tzik

Make symbolize.cc thread safe even on shared fds

ReadFromOffset in symbolize.cc used to call lseek() + read() to read
data from fd. However, the fd may be reused for multiple symbolize
requests from multiple threads, and causes a race around the fd read
offset.

This updates it to use pread() to resolve the race.
parent 5c292672
...@@ -136,17 +136,20 @@ _END_GOOGLE_NAMESPACE_ ...@@ -136,17 +136,20 @@ _END_GOOGLE_NAMESPACE_
_START_GOOGLE_NAMESPACE_ _START_GOOGLE_NAMESPACE_
// Read up to "count" bytes from file descriptor "fd" into the buffer // Read up to "count" bytes from "offset" in the file pointed by file
// starting at "buf" while handling short reads and EINTR. On // descriptor "fd" into the buffer starting at "buf" while handling short reads
// success, return the number of bytes read. Otherwise, return -1. // and EINTR. On success, return the number of bytes read. Otherwise, return
static ssize_t ReadPersistent(const int fd, void *buf, const size_t count) { // -1.
static ssize_t ReadFromOffset(const int fd, void *buf, const size_t count,
const off_t offset) {
SAFE_ASSERT(fd >= 0); SAFE_ASSERT(fd >= 0);
SAFE_ASSERT(count <= std::numeric_limits<ssize_t>::max()); SAFE_ASSERT(count <= std::numeric_limits<ssize_t>::max());
char *buf0 = reinterpret_cast<char *>(buf); char *buf0 = reinterpret_cast<char *>(buf);
ssize_t num_bytes = 0; ssize_t num_bytes = 0;
while (num_bytes < count) { while (num_bytes < count) {
ssize_t len; ssize_t len;
NO_INTR(len = read(fd, buf0 + num_bytes, count - num_bytes)); NO_INTR(len = pread(fd, buf0 + num_bytes, count - num_bytes,
offset + num_bytes));
if (len < 0) { // There was an error other than EINTR. if (len < 0) { // There was an error other than EINTR.
return -1; return -1;
} }
...@@ -159,18 +162,6 @@ static ssize_t ReadPersistent(const int fd, void *buf, const size_t count) { ...@@ -159,18 +162,6 @@ static ssize_t ReadPersistent(const int fd, void *buf, const size_t count) {
return num_bytes; return num_bytes;
} }
// Read up to "count" bytes from "offset" in the file pointed by file
// descriptor "fd" into the buffer starting at "buf". On success,
// return the number of bytes read. Otherwise, return -1.
static ssize_t ReadFromOffset(const int fd, void *buf,
const size_t count, const off_t offset) {
off_t off = lseek(fd, offset, SEEK_SET);
if (off == (off_t)-1) {
return -1;
}
return ReadPersistent(fd, buf, count);
}
// Try reading exactly "count" bytes from "offset" bytes in a file // Try reading exactly "count" bytes from "offset" bytes in a file
// pointed by "fd" into the buffer starting at "buf" while handling // pointed by "fd" into the buffer starting at "buf" while handling
// short reads and EINTR. On success, return true. Otherwise, return // short reads and EINTR. On success, return true. Otherwise, return
...@@ -398,9 +389,14 @@ struct FileDescriptor { ...@@ -398,9 +389,14 @@ struct FileDescriptor {
// and snprintf(). // and snprintf().
class LineReader { class LineReader {
public: public:
explicit LineReader(int fd, char *buf, int buf_len) : fd_(fd), explicit LineReader(int fd, char *buf, int buf_len, off_t offset)
buf_(buf), buf_len_(buf_len), bol_(buf), eol_(buf), eod_(buf) { : fd_(fd),
} buf_(buf),
buf_len_(buf_len),
offset_(offset),
bol_(buf),
eol_(buf),
eod_(buf) {}
// Read '\n'-terminated line from file. On success, modify "bol" // Read '\n'-terminated line from file. On success, modify "bol"
// and "eol", then return true. Otherwise, return false. // and "eol", then return true. Otherwise, return false.
...@@ -409,10 +405,11 @@ class LineReader { ...@@ -409,10 +405,11 @@ class LineReader {
// dropped. It's an intentional behavior to make the code simple. // dropped. It's an intentional behavior to make the code simple.
bool ReadLine(const char **bol, const char **eol) { bool ReadLine(const char **bol, const char **eol) {
if (BufferIsEmpty()) { // First time. if (BufferIsEmpty()) { // First time.
const ssize_t num_bytes = ReadPersistent(fd_, buf_, buf_len_); const ssize_t num_bytes = ReadFromOffset(fd_, buf_, buf_len_, offset_);
if (num_bytes <= 0) { // EOF or error. if (num_bytes <= 0) { // EOF or error.
return false; return false;
} }
offset_ += num_bytes;
eod_ = buf_ + num_bytes; eod_ = buf_ + num_bytes;
bol_ = buf_; bol_ = buf_;
} else { } else {
...@@ -425,11 +422,12 @@ class LineReader { ...@@ -425,11 +422,12 @@ class LineReader {
// Read text from file and append it. // Read text from file and append it.
char * const append_pos = buf_ + incomplete_line_length; char * const append_pos = buf_ + incomplete_line_length;
const int capacity_left = buf_len_ - incomplete_line_length; const int capacity_left = buf_len_ - incomplete_line_length;
const ssize_t num_bytes = ReadPersistent(fd_, append_pos, const ssize_t num_bytes =
capacity_left); ReadFromOffset(fd_, append_pos, capacity_left, offset_);
if (num_bytes <= 0) { // EOF or error. if (num_bytes <= 0) { // EOF or error.
return false; return false;
} }
offset_ += num_bytes;
eod_ = append_pos + num_bytes; eod_ = append_pos + num_bytes;
bol_ = buf_; bol_ = buf_;
} }
...@@ -474,6 +472,7 @@ class LineReader { ...@@ -474,6 +472,7 @@ class LineReader {
const int fd_; const int fd_;
char * const buf_; char * const buf_;
const int buf_len_; const int buf_len_;
off_t offset_;
char *bol_; char *bol_;
char *eol_; char *eol_;
const char *eod_; // End of data in "buf_". const char *eod_; // End of data in "buf_".
...@@ -532,7 +531,7 @@ OpenObjectFileContainingPcAndGetStartAddress(uint64_t pc, ...@@ -532,7 +531,7 @@ OpenObjectFileContainingPcAndGetStartAddress(uint64_t pc,
// look into the symbol tables inside. // look into the symbol tables inside.
char buf[1024]; // Big enough for line of sane /proc/self/maps char buf[1024]; // Big enough for line of sane /proc/self/maps
int num_maps = 0; int num_maps = 0;
LineReader reader(wrapped_maps_fd.get(), buf, sizeof(buf)); LineReader reader(wrapped_maps_fd.get(), buf, sizeof(buf), 0);
while (true) { while (true) {
num_maps++; num_maps++;
const char *cursor; const char *cursor;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment