// Copyright (c) 2010, 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 THE COPYRIGHT // OWNER 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. // Original author: Jim Blandy <jimb@mozilla.com> <jimb@red-bean.com> // cfi_frame_info.cc: Implementation of CFIFrameInfo class. // See cfi_frame_info.h for details. #include "processor/cfi_frame_info.h" #include <string.h> #include <sstream> #include "common/scoped_ptr.h" #include "processor/postfix_evaluator-inl.h" namespace google_breakpad { #ifdef _WIN32 #define strtok_r strtok_s #endif template<typename V> bool CFIFrameInfo::FindCallerRegs(const RegisterValueMap<V> ®isters, const MemoryRegion &memory, RegisterValueMap<V> *caller_registers) const { // If there are not rules for both .ra and .cfa in effect at this address, // don't use this CFI data for stack walking. if (cfa_rule_.empty() || ra_rule_.empty()) return false; RegisterValueMap<V> working; PostfixEvaluator<V> evaluator(&working, &memory); caller_registers->clear(); // First, compute the CFA. V cfa; working = registers; if (!evaluator.EvaluateForValue(cfa_rule_, &cfa)) return false; // Then, compute the return address. V ra; working = registers; working[".cfa"] = cfa; if (!evaluator.EvaluateForValue(ra_rule_, &ra)) return false; // Now, compute values for all the registers register_rules_ mentions. for (RuleMap::const_iterator it = register_rules_.begin(); it != register_rules_.end(); it++) { V value; working = registers; working[".cfa"] = cfa; if (!evaluator.EvaluateForValue(it->second, &value)) return false; (*caller_registers)[it->first] = value; } (*caller_registers)[".ra"] = ra; (*caller_registers)[".cfa"] = cfa; return true; } // Explicit instantiations for 32-bit and 64-bit architectures. template bool CFIFrameInfo::FindCallerRegs<uint32_t>( const RegisterValueMap<uint32_t> ®isters, const MemoryRegion &memory, RegisterValueMap<uint32_t> *caller_registers) const; template bool CFIFrameInfo::FindCallerRegs<uint64_t>( const RegisterValueMap<uint64_t> ®isters, const MemoryRegion &memory, RegisterValueMap<uint64_t> *caller_registers) const; string CFIFrameInfo::Serialize() const { std::ostringstream stream; if (!cfa_rule_.empty()) { stream << ".cfa: " << cfa_rule_; } if (!ra_rule_.empty()) { if (static_cast<std::streamoff>(stream.tellp()) != 0) stream << " "; stream << ".ra: " << ra_rule_; } for (RuleMap::const_iterator iter = register_rules_.begin(); iter != register_rules_.end(); ++iter) { if (static_cast<std::streamoff>(stream.tellp()) != 0) stream << " "; stream << iter->first << ": " << iter->second; } return stream.str(); } bool CFIRuleParser::Parse(const string &rule_set) { size_t rule_set_len = rule_set.size(); scoped_array<char> working_copy(new char[rule_set_len + 1]); memcpy(working_copy.get(), rule_set.data(), rule_set_len); working_copy[rule_set_len] = '\0'; name_.clear(); expression_.clear(); char *cursor; static const char token_breaks[] = " \t\r\n"; char *token = strtok_r(working_copy.get(), token_breaks, &cursor); for (;;) { // End of rule set? if (!token) return Report(); // Register/pseudoregister name? size_t token_len = strlen(token); if (token_len >= 1 && token[token_len - 1] == ':') { // Names can't be empty. if (token_len < 2) return false; // If there is any pending content, report it. if (!name_.empty() || !expression_.empty()) { if (!Report()) return false; } name_.assign(token, token_len - 1); expression_.clear(); } else { // Another expression component. assert(token_len > 0); // strtok_r guarantees this, I think. if (!expression_.empty()) expression_ += ' '; expression_ += token; } token = strtok_r(NULL, token_breaks, &cursor); } } bool CFIRuleParser::Report() { if (name_.empty() || expression_.empty()) return false; if (name_ == ".cfa") handler_->CFARule(expression_); else if (name_ == ".ra") handler_->RARule(expression_); else handler_->RegisterRule(name_, expression_); return true; } void CFIFrameInfoParseHandler::CFARule(const string &expression) { frame_info_->SetCFARule(expression); } void CFIFrameInfoParseHandler::RARule(const string &expression) { frame_info_->SetRARule(expression); } void CFIFrameInfoParseHandler::RegisterRule(const string &name, const string &expression) { frame_info_->SetRegisterRule(name, expression); } } // namespace google_breakpad