iOS: How to catch exception?

iOS: How to catch exception?

Table of Contents

1. System Crash

2. Processing signals

  • Here are some signal descriptions
  • Key points to note

3. Actual Combat

4. Crash Callstack Analysis – Further Analysis

5. Demo address

VI. References

Preface

Today in the iOS senior group, a friend asked about the problem of iOS exception capture. I have never studied this before, so I took this opportunity to study it. I also wrote a demo, which can be downloaded at the bottom of the article if needed.

Before reading this article, it is recommended that you read the article Talking about iOS Crash Collection Framework after reading this article to understand the principle.

When developing iOS applications, solving the Crash problem is always a difficult problem. There are two types of Crash. One is caused by EXC_BAD_ACCESS, which is caused by accessing a memory address that does not belong to the process, which may be accessing released memory; the other is an uncaught Objective-C exception (NSException), which causes the program to send a SIGABRT signal to itself and crash. In fact, for uncaught Objective-C exceptions, we have a way to record it. If the log is recorded properly, most crash problems can be solved. Here are the UI threads and background threads respectively.

1. System Crash

The abnormal exit of the program caused by system crash can be caught by the UncaughtExceptionHandler mechanism; that is, the content outside the catch in the program is caught by the system's built-in error handling. All we have to do is replace the ExceptionHandler with a custom function.

2. Processing signals

Using Objective-C's exception handling does not get signals. If we want to handle it, we also need to use the Unix standard signal mechanism to register processing functions for signals such as SIGABRT, SIGBUS, and SIGSEGV. In this function, we can output stack information, version information, and everything else we want.

Here are some signal descriptions

1) SIGHUP

This signal is sent when a user terminal connection ends (normally or abnormally), usually when the controlling process of the terminal ends, to notify the jobs in the same session that they are no longer associated with the controlling terminal.

When you log in to Linux, the system will assign a terminal (Session) to the logged-in user. All programs running in this terminal, including the foreground process group and the background process group, generally belong to this Session. When the user logs out of Linux, the foreground process group and the background process with terminal output will receive the SIGHUP signal. The default operation of this signal is to terminate the process, so the foreground process group and the background process with terminal output will be terminated. However, this signal can be captured. For example, wget can capture the SIGHUP signal and ignore it, so that even if you log out of Linux, wget can continue to download.

In addition, for a daemon that is disconnected from a terminal, this signal is used to notify it to re-read the configuration file.

2) SIGINT

Program termination (interrupt) signal, sent when the user types the INTR character (usually Ctrl-C), is used to notify the foreground process group to terminate the process.

3) SIGQUIT

Similar to SIGINT, but controlled by the QUIT character (usually Ctrl-). A process creates a core file when it exits due to SIGQUIT, which is similar to a program error signal in this sense.

4) SIGILL

An illegal instruction was executed. This is usually caused by an error in the executable file itself, or an attempt to execute a data segment. This signal may also be generated by a stack overflow.

5) SIGTRAP

Generated by a breakpoint instruction or other trap instruction. Used by the debugger.

6) SIGABRT

The signal generated by calling the abort function.

7) SIGBUS

Illegal address, including memory address alignment errors. For example, accessing a four-word integer, but its address is not a multiple of 4. It is different from SIGSEGV in that the latter is triggered by illegal access to a legal storage address (such as accessing non-own storage space or read-only storage space).

8) SIGFPE

Emitted when a fatal arithmetic error occurs. This includes not only floating point errors, but also overflows and division by zero and all other arithmetic errors.

9) SIGKILL

Used to terminate the program immediately. This signal cannot be blocked, processed or ignored. If the administrator finds that a process cannot be terminated, he can try to send this signal.

10) SIGUSR1

Leave it to the user

11) SIGSEGV

An attempt was made to access memory that was not allocated to the user, or an attempt was made to write data to a memory address that the user did not have permission to write to.

12) SIGUSR2

Leave it to the user

13) SIGPIPE

Pipeline Broken. This signal is usually generated in inter-process communication. For example, if two processes communicate using FIFO (pipe) and the reading pipe is not open or terminated unexpectedly, the writing process will receive the SIGPIPE signal. In addition, if two processes communicate using Socket, the reading process has terminated when the writing process writes to the Socket.

14) SIGALRM

Clock timing signal, which calculates the actual time or clock time. The alarm function uses this signal.

15) SIGTERM

The program terminate signal. Unlike SIGKILL, this signal can be blocked and processed. It is usually used to require the program to exit normally. The shell command kill generates this signal by default. If the process cannot be terminated, we will try SIGKILL.

17) SIGCHLD

When the child process ends, the parent process will receive this signal.

If the parent process does not handle this signal and does not wait for the child process, the child process will terminate but will still occupy an entry in the kernel process table. This child process is called a zombie process. We should avoid this situation (the parent process should either ignore the SIGCHILD signal, capture it, wait for its derived child process, or terminate the parent process first, in which case the termination of the child process will be automatically taken over by the init process).

18) SIGCONT

Allows a stopped process to continue execution. This signal cannot be blocked. A handler can be used to allow the program to complete specific work when it changes from the stopped state to the continued state. For example, redisplay the prompt

19) SIGSTOP

The execution of the process is stopped. Note the difference between it and terminate and interrupt: the process is not terminated, but just suspended. This signal cannot be blocked, processed or ignored.

20) SIGTSTP

Stops the process from running, but the signal can be handled and ignored. This signal is emitted when the user types the SUSP character (usually Ctrl-Z).

21) SIGTTIN

When a background job tries to read data from the user terminal, all processes in the job will receive a SIGTTIN signal. By default, these processes will stop executing.

22) SIGTTOU

Similar to SIGTTIN, but received when writing to the terminal (or changing the terminal mode).

23) SIGURG

Generated when "urgent" data or out-of-band data arrives at the socket.

24) SIGXCPU

CPU time resource limit exceeded. This limit can be read/changed with getrlimit/setrlimit.

25) SIGXFSZ

When a process attempts to enlarge a file so that it exceeds the file size resource limit.

26) SIGVTALRM

Virtual clock signal. Similar to SIGALRM, but it counts the CPU time used by the process.

27) SIGPROF

Similar to SIGALRM/SIGVTALRM, but includes CPU time used by the process as well as time spent in system calls.

28) SIGWINCH

Emitted when the window size changes.

29) SIGIO

The file descriptor is ready for input/output operations.

30) SIGPWR

Power failure

31) SIGSYS

Illegal system call.

Key points to note

  • Among the signals listed above, the signals that cannot be captured, blocked or ignored by the program are: SIGKILL, SIGSTOP
  • Signals that cannot be restored to default actions are: SIGILL, SIGTRAP
  • The default signals that cause process abortion are: SIGABRT, SIGBUS, SIGFPE, SIGILL, SIGIOT, SIGQUIT, SIGSEGV, SIGTRAP, SIGXCPU, SIGXFSZ
  • The default signals that cause the process to exit are:
  • SIGALRM,SIGHUP,SIGINT,SIGKILL,SIGPIPE,SIGPOLL,SIGPROF,SIGSYS,SIGTERM,SIGUSR1,SIGUSR2,SIGVTALRM
  • The default signals that will cause the process to stop are: SIGSTOP, SIGTSTP, SIGTTIN, SIGTTOU
  • The signals ignored by the default process are: SIGCHLD, SIGPWR, SIGURG, SIGWINCH
  • In addition, SIGIO exits in SVR4 and is ignored in 4.3BSD; SIGCONT continues when the process is suspended, otherwise it is ignored and cannot be blocked.

3. Actual Combat

1. AppDelegate.m

  1. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  2.  
  3. // Override point for customization after application launch.
  4.  
  5.      
  6.  
  7. InstallSignalHandler(); //Signal truncation
  8.  
  9. InstallUncaughtExceptionHandler(); //System exception capture
  10.  
  11.      
  12.  
  13. return YES;
  14.  
  15. }

2. Implementation of SignalHandler.m

  1. void SignalExceptionHandler( int signal)
  2.  
  3. {
  4.  
  5. NSMutableString *mstr = [[NSMutableString alloc] init];
  6.  
  7. [mstr appendString:@ "Stack:\n" ];
  8.  
  9. void* callstack[128];
  10.  
  11. int i, frames = backtrace(callstack, 128);
  12.  
  13. char ** strs = backtrace_symbols(callstack, frames);
  14.  
  15. for (i = 0; i
  16.  
  17. [mstr appendFormat:@ "%s\n" , strs[i]];
  18.  
  19. }
  20.  
  21. [SignalHandler saveCreash:mstr];
  22.  
  23.   
  24.  
  25. }
  26.  
  27.   
  28.  
  29. void InstallSignalHandler(void)
  30.  
  31. {
  32.  
  33. signal(SIGHUP, SignalExceptionHandler);
  34.  
  35. signal(SIGINT, SignalExceptionHandler);
  36.  
  37. signal(SIGQUIT, SignalExceptionHandler);
  38.  
  39.      
  40.  
  41. signal(SIGABRT, SignalExceptionHandler);
  42.  
  43. signal(SIGILL, SignalExceptionHandler);
  44.  
  45. signal(SIGSEGV, SignalExceptionHandler);
  46.  
  47. signal(SIGFPE, SignalExceptionHandler);
  48.  
  49. signal(SIGBUS, SignalExceptionHandler);
  50.  
  51. signal(SIGPIPE, SignalExceptionHandler);
  52.  
  53. }

For more information about error types, see the description above. SignalExceptionHandler is the callback for signal errors. When a signal error occurs, you can call back to this method.

3. Implementation of UncaughtExceptionHandler.m

  1. void HandleException(NSException *exception)
  2.  
  3. {
  4.  
  5. // Exception stack information
  6.  
  7. NSArray *stackArray = [exception callStackSymbols];
  8.  
  9. // Cause of the exception
  10.  
  11. NSString *reason = [exception reason];
  12.  
  13. // Exception name
  14.  
  15. NSString * name = [exception name ];
  16.  
  17. NSString *exceptionInfo = [NSString stringWithFormat:@ "Exception reason: %@\nException name: %@\nException stack: %@" , name , reason, stackArray];
  18.  
  19. NSLog(@ "%@" , exceptionInfo);
  20.  
  21. [UncaughtExceptionHandler saveCreash:exceptionInfo];
  22.  
  23. }
  24.  
  25.   
  26.  
  27. void InstallUncaughtExceptionHandler(void)
  28.  
  29. {
  30.  
  31. NSSetUncaughtExceptionHandler(&HandleException);
  32.  
  33. }

4. Testing – the key to avoiding pitfalls

The most critical step here is not to test SignalHandler in debug environment. Because the system debug will intercept first. We need to run it once and then turn off debug status. We should directly click on the app we built on the simulator to run it. UncaughtExceptionHandler can capture it in debug mode.

  1. - (IBAction)buttonClick:(UIButton *)sender {
  2.  
  3. //1.Semaphore
  4.  
  5. Test *pTest = {1,2};
  6.  
  7. free (pTest); // Causes SIGABRT error, because there is no such space in the memory, where does the free come from ? It is just an object in the stack.
  8.  
  9. pTest->a = 5;
  10.  
  11. }
  12.  
  13. - (IBAction)buttonOCException:(UIButton *)sender
  14.  
  15. {
  16.  
  17. //2.ios crash
  18.  
  19. NSArray *array= @[@ "tom" ,@ "xxx" ,@ "ooo" ];
  20.  
  21. [array objectAtIndex:5];
  22.  
  23. }

4. Crash Callstack Analysis – Further Analysis

property illustrate
0x8badf00d Taking too long to start, terminate an application, or respond to system events, meaning "eating bad food."
0xdeadfa11 The user forced exit, which means "dead fall". (When the system is unresponsive, the user presses the power switch and HOME)
0xbaaaaaad The user holds down the Home button and the volume button to get the current memory status, which does not mean a crash.
0xbad22222 VoIP application crashes due to frequent recovery
0xc00010ff Because it was too hot, it was dried, meaning "cool off"
0xdead10cc Because it still occupies system resources (such as the address book) in the background, it is killed, which means "dead lock"

5. Demo address

iOSCrashUncaught Download

https://github.com/xcysuccess/iOSCrashUncaught

VI. References

1. Debugging skills after program crash

2. The problem of iOS socket program being terminated by SIGPIPE signal

3. Beautiful Nian Xi

4. How to locate random crashes of wild pointers in Obj-C (I): First, increase the wild pointer crash rate

5. How to locate random crashes in Obj-C wild pointers (Part 2): Make non-necessary crashes mandatory

6. How to locate the random crash of Obj-C wild pointer (Part 3): Add some black technology to make the crash reveal itself

<<:  Aite Tribe Story Collection (12): Habits lead to skill improvement

>>:  An Android client architecture design sharing

Recommend

2019, the year marketers “swim naked”

Everyone knows that my writing is usually very dr...

How long can a pair of self-heating insoles last?

As we all know, the cold starts from the feet, es...

How can Douyin increase followers quickly and effectively?

There are three ways to increase followers: conte...

How to efficiently guide products to achieve self-propagation growth model

As a member of the growth department, when observ...

How to invest in Baidu bidding and CPC in 2021?

We all know that now o cpc has become the mainstr...

The calm before the outbreak? Comprehensive thinking about "Mini Programs"

Since WeChat announced the upcoming release of Mi...

How to find more than 200 channels in App promotion

Start from seven aspects including website naviga...

Summary of the ten core advantages of HTML 5

[51CTO.com Quick Translation] What is the most po...

How to plan an event with high user participation?

0. What is event operation? Before we talk about ...

Five paths to building a TOB brand

When I was thinking about the issues of TOB brand...