| // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "base/build/build_config.h" | 
 |  | 
 | #if defined(COMPILER_MSVC) | 
 | #include <intrin.h> | 
 | #endif | 
 |  | 
 | #include "base/location.h" | 
 | #include "base/strings/string_number_conversions.h" | 
 | #include "base/strings/stringprintf.h" | 
 |  | 
 | namespace tracked_objects { | 
 |  | 
 | Location::Location(const char* function_name, | 
 |                    const char* file_name, | 
 |                    int line_number, | 
 |                    const void* program_counter) | 
 |     : function_name_(function_name), | 
 |       file_name_(file_name), | 
 |       line_number_(line_number), | 
 |       program_counter_(program_counter) { | 
 | } | 
 |  | 
 | Location::Location() | 
 |     : function_name_("Unknown"), | 
 |       file_name_("Unknown"), | 
 |       line_number_(-1), | 
 |       program_counter_(NULL) { | 
 | } | 
 |  | 
 | Location::Location(const Location& other) | 
 |     : function_name_(other.function_name_), | 
 |       file_name_(other.file_name_), | 
 |       line_number_(other.line_number_), | 
 |       program_counter_(other.program_counter_) { | 
 | } | 
 |  | 
 | std::string Location::ToString() const { | 
 |   return std::string(function_name_) + "@" + file_name_ + ":" + | 
 |       base::IntToString(line_number_); | 
 | } | 
 |  | 
 | void Location::Write(bool display_filename, bool display_function_name, | 
 |                      std::string* output) const { | 
 |   base::StringAppendF(output, "%s[%d] ", | 
 |       display_filename ? file_name_ : "line", | 
 |       line_number_); | 
 |  | 
 |   if (display_function_name) { | 
 |     WriteFunctionName(output); | 
 |     output->push_back(' '); | 
 |   } | 
 | } | 
 |  | 
 | void Location::WriteFunctionName(std::string* output) const { | 
 |   // Translate "<" to "<" for HTML safety. | 
 |   // TODO(jar): Support ASCII or html for logging in ASCII. | 
 |   for (const char *p = function_name_; *p; p++) { | 
 |     switch (*p) { | 
 |       case '<': | 
 |         output->append("<"); | 
 |         break; | 
 |  | 
 |       case '>': | 
 |         output->append(">"); | 
 |         break; | 
 |  | 
 |       default: | 
 |         output->push_back(*p); | 
 |         break; | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | //------------------------------------------------------------------------------ | 
 | LocationSnapshot::LocationSnapshot() : line_number(-1) { | 
 | } | 
 |  | 
 | LocationSnapshot::LocationSnapshot( | 
 |     const tracked_objects::Location& location) | 
 |     : file_name(location.file_name()), | 
 |       function_name(location.function_name()), | 
 |       line_number(location.line_number()) { | 
 | } | 
 |  | 
 | LocationSnapshot::~LocationSnapshot() { | 
 | } | 
 |  | 
 | //------------------------------------------------------------------------------ | 
 | #if defined(COMPILER_MSVC) | 
 | __declspec(noinline) | 
 | #endif | 
 |     const void* GetProgramCounter() { | 
 | #if defined(COMPILER_MSVC) | 
 |   return _ReturnAddress(); | 
 | #elif defined(COMPILER_GCC) && !defined(OS_NACL) | 
 |   return __builtin_extract_return_addr(__builtin_return_address(0)); | 
 | #else | 
 |   return NULL; | 
 | #endif | 
 | } | 
 |  | 
 | }  // namespace tracked_objects |