LCOV - code coverage report
Current view: top level - toolkit/crashreporter/breakpad-client/linux/handler - exception_handler.h (source / functions) Hit Total Coverage
Test: output.info Lines: 0 7 0.0 %
Date: 2017-07-14 16:53:18 Functions: 0 3 0.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : // Copyright (c) 2010 Google Inc.
       2             : // All rights reserved.
       3             : //
       4             : // Redistribution and use in source and binary forms, with or without
       5             : // modification, are permitted provided that the following conditions are
       6             : // met:
       7             : //
       8             : //     * Redistributions of source code must retain the above copyright
       9             : // notice, this list of conditions and the following disclaimer.
      10             : //     * Redistributions in binary form must reproduce the above
      11             : // copyright notice, this list of conditions and the following disclaimer
      12             : // in the documentation and/or other materials provided with the
      13             : // distribution.
      14             : //     * Neither the name of Google Inc. nor the names of its
      15             : // contributors may be used to endorse or promote products derived from
      16             : // this software without specific prior written permission.
      17             : //
      18             : // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
      19             : // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
      20             : // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
      21             : // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
      22             : // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
      23             : // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
      24             : // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
      25             : // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
      26             : // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
      27             : // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
      28             : // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
      29             : 
      30             : #ifndef CLIENT_LINUX_HANDLER_EXCEPTION_HANDLER_H_
      31             : #define CLIENT_LINUX_HANDLER_EXCEPTION_HANDLER_H_
      32             : 
      33             : #include <signal.h>
      34             : #include <stdint.h>
      35             : #include <stdio.h>
      36             : #include <sys/ucontext.h>
      37             : 
      38             : #include <string>
      39             : 
      40             : #include "linux/crash_generation/crash_generation_client.h"
      41             : #include "linux/handler/minidump_descriptor.h"
      42             : #include "linux/minidump_writer/minidump_writer.h"
      43             : #include "common/scoped_ptr.h"
      44             : #include "common/using_std_string.h"
      45             : #include "google_breakpad/common/minidump_format.h"
      46             : 
      47             : namespace google_breakpad {
      48             : 
      49             : // ExceptionHandler
      50             : //
      51             : // ExceptionHandler can write a minidump file when an exception occurs,
      52             : // or when WriteMinidump() is called explicitly by your program.
      53             : //
      54             : // To have the exception handler write minidumps when an uncaught exception
      55             : // (crash) occurs, you should create an instance early in the execution
      56             : // of your program, and keep it around for the entire time you want to
      57             : // have crash handling active (typically, until shutdown).
      58             : // (NOTE): There should be only be one this kind of exception handler
      59             : // object per process.
      60             : //
      61             : // If you want to write minidumps without installing the exception handler,
      62             : // you can create an ExceptionHandler with install_handler set to false,
      63             : // then call WriteMinidump.  You can also use this technique if you want to
      64             : // use different minidump callbacks for different call sites.
      65             : //
      66             : // In either case, a callback function is called when a minidump is written,
      67             : // which receives the full path or file descriptor of the minidump.  The
      68             : // caller can collect and write additional application state to that minidump,
      69             : // and launch an external crash-reporting application.
      70             : //
      71             : // Caller should try to make the callbacks as crash-friendly as possible,
      72             : // it should avoid use heap memory allocation as much as possible.
      73             : 
      74             : class ExceptionHandler {
      75             :  public:
      76             :   // A callback function to run before Breakpad performs any substantial
      77             :   // processing of an exception.  A FilterCallback is called before writing
      78             :   // a minidump.  |context| is the parameter supplied by the user as
      79             :   // callback_context when the handler was created.
      80             :   //
      81             :   // If a FilterCallback returns true, Breakpad will continue processing,
      82             :   // attempting to write a minidump.  If a FilterCallback returns false,
      83             :   // Breakpad  will immediately report the exception as unhandled without
      84             :   // writing a minidump, allowing another handler the opportunity to handle it.
      85             :   typedef bool (*FilterCallback)(void *context);
      86             : 
      87             :   // A callback function to run after the minidump has been written.
      88             :   // |descriptor| contains the file descriptor or file path containing the
      89             :   // minidump. |context| is the parameter supplied by the user as
      90             :   // callback_context when the handler was created.  |succeeded| indicates
      91             :   // whether a minidump file was successfully written.
      92             :   //
      93             :   // If an exception occurred and the callback returns true, Breakpad will
      94             :   // treat the exception as fully-handled, suppressing any other handlers from
      95             :   // being notified of the exception.  If the callback returns false, Breakpad
      96             :   // will treat the exception as unhandled, and allow another handler to handle
      97             :   // it. If there are no other handlers, Breakpad will report the exception to
      98             :   // the system as unhandled, allowing a debugger or native crash dialog the
      99             :   // opportunity to handle the exception.  Most callback implementations
     100             :   // should normally return the value of |succeeded|, or when they wish to
     101             :   // not report an exception of handled, false.  Callbacks will rarely want to
     102             :   // return true directly (unless |succeeded| is true).
     103             :   typedef bool (*MinidumpCallback)(const MinidumpDescriptor& descriptor,
     104             :                                    void* context,
     105             :                                    bool succeeded);
     106             : 
     107             :   // In certain cases, a user may wish to handle the generation of the minidump
     108             :   // themselves. In this case, they can install a handler callback which is
     109             :   // called when a crash has occurred. If this function returns true, no other
     110             :   // processing of occurs and the process will shortly be crashed. If this
     111             :   // returns false, the normal processing continues.
     112             :   typedef bool (*HandlerCallback)(const void* crash_context,
     113             :                                   size_t crash_context_size,
     114             :                                   void* context);
     115             : 
     116             :   // Creates a new ExceptionHandler instance to handle writing minidumps.
     117             :   // Before writing a minidump, the optional |filter| callback will be called.
     118             :   // Its return value determines whether or not Breakpad should write a
     119             :   // minidump.  The minidump content will be written to the file path or file
     120             :   // descriptor from |descriptor|, and the optional |callback| is called after
     121             :   // writing the dump file, as described above.
     122             :   // If install_handler is true, then a minidump will be written whenever
     123             :   // an unhandled exception occurs.  If it is false, minidumps will only
     124             :   // be written when WriteMinidump is called.
     125             :   // If |server_fd| is valid, the minidump is generated out-of-process.  If it
     126             :   // is -1, in-process generation will always be used.
     127             :   ExceptionHandler(const MinidumpDescriptor& descriptor,
     128             :                    FilterCallback filter,
     129             :                    MinidumpCallback callback,
     130             :                    void* callback_context,
     131             :                    bool install_handler,
     132             :                    const int server_fd);
     133             :   ~ExceptionHandler();
     134             : 
     135           0 :   const MinidumpDescriptor& minidump_descriptor() const {
     136           0 :     return minidump_descriptor_;
     137             :   }
     138             : 
     139           0 :   void set_minidump_descriptor(const MinidumpDescriptor& descriptor) {
     140           0 :     minidump_descriptor_ = descriptor;
     141           0 :   }
     142             : 
     143             :   void set_crash_handler(HandlerCallback callback) {
     144             :     crash_handler_ = callback;
     145             :   }
     146             : 
     147             :   void set_crash_generation_client(CrashGenerationClient* client) {
     148             :     crash_generation_client_.reset(client);
     149             :   }
     150             : 
     151             :   // Writes a minidump immediately.  This can be used to capture the execution
     152             :   // state independently of a crash.
     153             :   // Returns true on success.
     154             :   // If the ExceptionHandler has been created with a path, a new file is
     155             :   // generated for each minidump.  The file path can be retrieved in the
     156             :   // MinidumpDescriptor passed to the MinidumpCallback or by accessing the
     157             :   // MinidumpDescriptor directly from the ExceptionHandler (with
     158             :   // minidump_descriptor()).
     159             :   // If the ExceptionHandler has been created with a file descriptor, the file
     160             :   // descriptor is repositioned to its beginning and the previous generated
     161             :   // minidump is overwritten.
     162             :   // Note that this method is not supposed to be called from a compromised
     163             :   // context as it uses the heap.
     164             :   bool WriteMinidump();
     165             : 
     166             :   // Convenience form of WriteMinidump which does not require an
     167             :   // ExceptionHandler instance.
     168             :   static bool WriteMinidump(const string& dump_path,
     169             :                             MinidumpCallback callback,
     170             :                             void* callback_context);
     171             : 
     172             :   // Write a minidump of |child| immediately.  This can be used to
     173             :   // capture the execution state of |child| independently of a crash.
     174             :   // Pass a meaningful |child_blamed_thread| to make that thread in
     175             :   // the child process the one from which a crash signature is
     176             :   // extracted.
     177             :   //
     178             :   // WARNING: the return of this function *must* happen before
     179             :   // the code that will eventually reap |child| executes.
     180             :   // Otherwise there's a pernicious race condition in which |child|
     181             :   // exits, is reaped, another process created with its pid, then that
     182             :   // new process dumped.
     183             :   static bool WriteMinidumpForChild(pid_t child,
     184             :                                     pid_t child_blamed_thread,
     185             :                                     const string& dump_path,
     186             :                                     MinidumpCallback callback,
     187             :                                     void* callback_context);
     188             : 
     189             :   // This structure is passed to minidump_writer.h:WriteMinidump via an opaque
     190             :   // blob. It shouldn't be needed in any user code.
     191             :   struct CrashContext {
     192             :     siginfo_t siginfo;
     193             :     pid_t tid;  // the crashing thread.
     194             :     struct ucontext context;
     195             : #if !defined(__ARM_EABI__) && !defined(__mips__)
     196             :     // #ifdef this out because FP state is not part of user ABI for Linux ARM.
     197             :     // In case of MIPS Linux FP state is already part of struct
     198             :     // ucontext so 'float_state' is not required.
     199             :     fpstate_t float_state;
     200             : #endif
     201             :   };
     202             : 
     203             :   // Returns whether out-of-process dump generation is used or not.
     204           0 :   bool IsOutOfProcess() const {
     205           0 :     return crash_generation_client_.get() != NULL;
     206             :   }
     207             : 
     208             :   // Add information about a memory mapping. This can be used if
     209             :   // a custom library loader is used that maps things in a way
     210             :   // that the linux dumper can't handle by reading the maps file.
     211             :   void AddMappingInfo(const string& name,
     212             :                       const uint8_t identifier[sizeof(MDGUID)],
     213             :                       uintptr_t start_address,
     214             :                       size_t mapping_size,
     215             :                       size_t file_offset);
     216             : 
     217             :   // Register a block of memory of length bytes starting at address ptr
     218             :   // to be copied to the minidump when a crash happens.
     219             :   void RegisterAppMemory(void* ptr, size_t length);
     220             : 
     221             :   // Unregister a block of memory that was registered with RegisterAppMemory.
     222             :   void UnregisterAppMemory(void* ptr);
     223             : 
     224             :   // Force signal handling for the specified signal.
     225             :   bool SimulateSignalDelivery(int sig);
     226             : 
     227             :   // Report a crash signal from an SA_SIGINFO signal handler.
     228             :   bool HandleSignal(int sig, siginfo_t* info, void* uc);
     229             : 
     230             :  private:
     231             :   // Save the old signal handlers and install new ones.
     232             :   static bool InstallHandlersLocked();
     233             :   // Restore the old signal handlers.
     234             :   static void RestoreHandlersLocked();
     235             : 
     236             :   void PreresolveSymbols();
     237             :   bool GenerateDump(CrashContext *context);
     238             :   void SendContinueSignalToChild();
     239             :   void WaitForContinueSignal();
     240             : 
     241             :   static void SignalHandler(int sig, siginfo_t* info, void* uc);
     242             :   static int ThreadEntry(void* arg);
     243             :   bool DoDump(pid_t crashing_process, const void* context,
     244             :               size_t context_size);
     245             : 
     246             :   const FilterCallback filter_;
     247             :   const MinidumpCallback callback_;
     248             :   void* const callback_context_;
     249             : 
     250             :   scoped_ptr<CrashGenerationClient> crash_generation_client_;
     251             : 
     252             :   MinidumpDescriptor minidump_descriptor_;
     253             : 
     254             :   // Must be volatile. The compiler is unaware of the code which runs in
     255             :   // the signal handler which reads this variable. Without volatile the
     256             :   // compiler is free to optimise away writes to this variable which it
     257             :   // believes are never read.
     258             :   volatile HandlerCallback crash_handler_;
     259             : 
     260             :   // We need to explicitly enable ptrace of parent processes on some
     261             :   // kernels, but we need to know the PID of the cloned process before we
     262             :   // can do this. We create a pipe which we can use to block the
     263             :   // cloned process after creating it, until we have explicitly enabled
     264             :   // ptrace. This is used to store the file descriptors for the pipe
     265             :   int fdes[2];
     266             : 
     267             :   // Callers can add extra info about mappings for cases where the
     268             :   // dumper code cannot extract enough information from /proc/<pid>/maps.
     269             :   MappingList mapping_list_;
     270             : 
     271             :   // Callers can request additional memory regions to be included in
     272             :   // the dump.
     273             :   AppMemoryList app_memory_list_;
     274             : };
     275             : 
     276             : }  // namespace google_breakpad
     277             : 
     278             : #endif  // CLIENT_LINUX_HANDLER_EXCEPTION_HANDLER_H_

Generated by: LCOV version 1.13