As developers, we spend 70% of our time debugging. 20% of the time is spent on architectural design and team communication, and only 10% is spent writing code.
So it’s important to make that 70% of the time productive. The LLDB is a good choice. Although Xcode has a full-featured debug panel, the console is still an important part of debugging. Next I’ll explore some of the LLDB debugging techniques that are commonly used at work.
Where to start?
The LLDB has a number of useful debugging tools. I’ll pick a few important commands to discuss:
- Get variable values:
expression
.e
.print
.po
.p
- Get execution environment + language-specific command:
bugreport
.frame
.language
- Implement process control:
process
.breakpoint
.thread
.watchpoint
- Other:
command
.platform
.gui
The following lists descriptions and examples of common LLDB commands. You can save the picture for future reference.
1. Obtain variable values and status
Commands: expression, e, print, Po, p
The most basic function of a debugging tool is to print and modify the value of a variable. Expression or E is such a tool. You can execute almost any expression or command at run time.
Suppose you are debugging the valueOfLifeWithoutSumOf() method, which adds two numbers and subtracts from 42.
Now the run has the wrong result. You can modify the code to try to locate the problem as shown below:
A better way is to use the expression command to change the value of a variable at run time. Set a breakpoint and run.
Print the value of the variable in LLDB format:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) e <variable>__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The expression is executed in exactly the same way:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) e <expression>__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) e sum (Int) $R0 = 6 // $R0 (LLDB) e sum = 4 (LLDB) e sum (Int) $R2 = 4 (LLDB) e sum (Int) $R2 = 4 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The expression command has several options. LLDB uses double dashes — to separate options and expressions:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) expression <some flags> -- <variable>__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Expression has about 30 options. These options are worth exploring. To obtain detailed documents, run the following command:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__> lldb
> (lldb) help # To explore all available commands
> (lldb) help expression # To explore all expressions related sub-commands
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Some of the more common options are listed below:
-D <count>
(--depth <count>
) – Sets the recursion depth of print aggregation types (default is infinite recursion).-O
(--object-desctiption
) – Prints the description method.-T
(--show-types
) – Displays the type of each variable.-f <format>
(--format<format>
) – Sets the output format.-i <boolean>
(--ignore-breakpoints <boolean>
) – Ignore breakpoints in an expression while running it.
Let’s say I have a Logger object. This object has strings and structs as properties. If you want to print only level 1 properties, use the -d option:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) e -D 1 -- logger (LLDB_Debugger_Exploration.Logger) $R5 = 0x0000608000087e90 { currentClassName = "ViewController" debuggerStruct ={... } } __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
By default, LLDB recursively prints all attributes of an object, showing very detailed content:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) e -- logger
(LLDB_Debugger_Exploration.Logger) $R6 = 0x0000608000087e90 {
currentClassName = "ViewController"
debuggerStruct = (methodName = "name", lineNumber = 2, commandCounter = 23)
}
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
It can also be printed using e-o — or alias Po as follows:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) po logger
<Logger: 0x608000087e90>__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
It’s not very readable to print directly. By implementing CustomStringConvertible var Description: String {return… } property, Po returns a more readable description.
The print command is also mentioned at the beginning of this paragraph. Print
is almost the same as expression —
, except that the print command has no options available and does not need to pass arguments.
2. Obtain the execution environment and commands in a specific language
Bugreport, Frame, language
Do you often need to copy and paste logs into task manager to locate problems? The bugreport command of LLDB generates a detailed report of the current state of the app. This command is useful when you want to delay tracking down problems. To save the status of your app, you can use Bugreport to generate reports.
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) bugreport unwind --outfile <path to output file>__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The generated report is as follows:
! Example of bugreport command output
The frame command prints the stack frame of the current thread:
Use the following command to quickly understand where you are and the current execution environment:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) frame info
frame #0: 0x000000010bbe4b4d LLDB-Debugger-Exploration`ViewController.valueOfLifeWithoutSumOf(a=2, b=2, self=0x00007fa0c1406900) -> Int at ViewController.swift:96
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
This information is useful for breakpoint management as discussed later in this article.
The LLDB has language-specific commands. These are for C++, Objective-C, Swift, and RenderScript. Only two commands for Swift are discussed in this article: demangle and refcount.
The demangle command is used to fix the corrupted Swift type name (created during compilation to avoid namespace problems) as its name suggests. WWDC14 session – “Advanced Swift Debugging in LLDB”
The refcount command is also intuitively named to display the reference count of an object. Let’s go back to the output object we discussed earlier – Logger:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) language swift refcount logger
refcount data: (strong = 4, weak = 0)
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
This is very helpful for debugging to find memory leak problems.
3. Implement process control
Process, breakpoint, thread
This is my favorite part. Using these commands from LLDB (especially Breakpoint) makes debugging more automated. Helps speed up debugging.
Process is used to control the process to be debugged, and the LLDB can be attached to or unbound from a specific target. Xcode did the LLDB attaching to the process for us when we ran target, so WE won’t talk about LLDB attaching here. Read the Apple manual -“Using LLDB as a Standalone Debugger” to learn how to attach the LLDB to a process.
Process Status prints information about the current process and breakpoint:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) process status
Process 27408 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = step over
frame #0: 0x000000010bbe4889 LLDB-Debugger-Exploration`ViewController.viewDidLoad(self=0x00007fa0c1406900) -> () at ViewController.swift:69
66
67 let a = 2, b = 2
68 let result = valueOfLifeWithoutSumOf(a, and: b)
-> 69 print(result)
70
71
72
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
To continue, run the following command:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) process continue (lldb) c // This is equivalent to the command __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
This is equivalent to the “Continue” button in the Xcode debug panel:
Breakpoint is used for various operations on breakpoints. These all-too-common commands: breakpoint enable, breakpoint disable, and breakpoint delete won’t be discussed here.
All breakpoints can be printed using the list subcommand:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) breakpoint list Current breakpoints: 1: file = '/Users/Ahmed/Desktop/Recent/LLDB-Debugger-Exploration/LLDB-Debugger-Exploration/ViewController.swift', Line = 95, exact_match = 0, locations = 1, Resolved = 1, Hit count = 1 where = LLDB-Debugger-Exploration`LLDB_Debugger_Exploration.ViewController.valueOfLifeWithoutSumOf (Swift.Int, and : Swift.Int) -> Swift.Int + 27 at ViewController.swift:95, address = 0x0000000107f3eb3b, resolved, hit count = 1 2: file = '/Users/Ahmed/Desktop/Recent/LLDB-Debugger-Exploration/LLDB-Debugger-Exploration/ViewController.swift', Line = 60, exact_match = 0, locations = 1, Resolved = 1, Hit count = 1 where = LLDB-Debugger-Exploration`LLDB_Debugger_Exploration.ViewController.viewDidLoad () -> () + 521 at ViewController.swift:60, address = 0x0000000107f3e609, resolved, hit count = 1 __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The first number in the list is the breakpoint ID, which can be used to reference the corresponding breakpoint. Set a new breakpoint from the console:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) breakpoint set -f ViewController.swift -l 96
Breakpoint 3: where = LLDB-Debugger-Exploration`LLDB_Debugger_Exploration.ViewController.valueOfLifeWithoutSumOf (Swift.Int, and : Swift.Int) -> Swift.Int + 45 at ViewController.swift:96, address = 0x0000000107f3eb4d
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The -f option in the above example is used to specify the file in which the breakpoint is located. The -l option specifies the number of lines on which the breakpoint is located. Here’s a more concise way to express the above example:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) b ViewController.swift:96__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
You can also set a breakpoint on a method using a command:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) breakpoint set --one-shot Swift -l 90 (LLDB) br s -o -f ViewController. Swift -l 91 // A condensed version of __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Sometimes you need a breakpoint to hit only once. Delete the breakpoint immediately after it is hit. You can run the following command:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) breakpoint set --one-shot Swift -l 90 (LLDB) br s -o -f ViewController. Swift -l 91 // A condensed version of __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Now comes the most interesting part – breakpoint automation. Did you know that you can set certain actions to be executed when a breakpoint occurs? When debugging, do you like to use print() in your code to print the values you are interested in? Here’s a better way to do it.
Breakpoint Command allows you to set the command to be executed after a breakpoint is hit. You can even set “transparent” breakpoints that do not interrupt execution. Technically, “transparent” breakpoints already break execution, but they can be made insensitive by adding the continue command to the chain of command.
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) b ViewController.swift:96 // 设置断点
Breakpoint 2: where = LLDB-Debugger-Exploration`LLDB_Debugger_Exploration.ViewController.valueOfLifeWithoutSumOf (Swift.Int, and : Swift.Int) -> Swift.Int + 45 at ViewController.swift:96, address = 0x000000010c555b4d
(lldb) breakpoint command add 2 // Setup some commands
Enter your debugger command(s). Type 'DONE' to end.
> p sum // Print value of "sum" variable
> p a + b // Evaluate a + b
> DONE
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Breakpoint command list
to check whether all the commands are correct:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) breakpoint command list 2
Breakpoint 2:
Breakpoint commands:
p sum
p a + b
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
When a breakpoint hits, the console displays the following output:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Process 36612 resuming p sum (Int) $R0 = 6 p a + b (Int) $R1 = 4 __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
This is exactly what we expected. You can even add continue to the end of the chain of command to prevent execution from being interrupted.
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) breakpoint command add 2 // Setup some commands Enter your debugger command(s). Type 'DONE' to end. > p sum // Print value of "sum" variable > p a + b // Evaluate a + b > continue // Resume right after first hit > DONE __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Output result:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__p sum
(Int) $R0 = 6
p a + b
(Int) $R1 = 4
continue
Process 36863 resuming
Command #3 'continue' continued the target.
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The execution process can be completely controlled by thread and its sub-commands: step-over, step-in, step-out, and continue. These commands are equivalent to the buttons in the Xcode debug panel that control the execution flow.
There are also predefined shortcut commands for these commands:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) thread step-over (lldb) Next // Equals to "Thread step-over" (LLDB) n // Equals to "next" (LLDB) Thread step-in (LLDB) step // Equals to "thread step-in" (LLDB) s // Equivalent to "step" __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
To obtain more information about the current thread, use the info subcommand:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) thread info
thread #1: tid = 0x17de17, 0x0000000109429a90 LLDB-Debugger-Exploration`ViewController.sumOf(a=2, b=2, self=0x00007fe775507390) -> Int at ViewController.swift:90, queue = 'com.apple.main-thread', stop reason = step i
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The list subcommand is used to display all active threads:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) thread list
Process 50693 stopped
* thread #1: tid = 0x17de17, 0x0000000109429a90 LLDB-Debugger-Exploration`ViewController.sumOf(a=2, b=2, self=0x00007fe775507390) -> Int at ViewController.swift:90, queue = 'com.apple.main-thread', stop reason = step in
thread #2: tid = 0x17df4a, 0x000000010daa4dc6 libsystem_kernel.dylib`kevent_qos + 10, queue = 'com.apple.libdispatch-manager'
thread #3: tid = 0x17df4b, 0x000000010daa444e libsystem_kernel.dylib`__workq_kernreturn + 10
thread #5: tid = 0x17df4e, 0x000000010da9c34a libsystem_kernel.dylib`mach_msg_trap + 10, name = 'com.apple.uikit.eventfetch-thread'
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
other
Command, Platform, GUI
The LLDB has a command for managing other commands. It sounds strange, but it’s very useful. First, this command allows you to execute LLDB commands from a file. So you can create a script that contains a lot of useful commands and execute it as a command. Here is a file with two commands:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__thread info // Displays the current thread information br list // Display all breakpoints __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The actual commands are as follows:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) command source /Users/Ahmed/Desktop/lldb-test-script Executing commands in '/Users/Ahmed/Desktop/lldb-test-script'. thread info thread # 1: tid = 0x17de17, 0x0000000109429a90 LLDB-Debugger-Exploration`ViewController.sumOf(a=2, b=2, self=0x00007fe775507390) -> Int at ViewController.swift:90, queue = 'com.apple.main-thread', stop reason = step in br list Current breakpoints: 1: file = '/Users/Ahmed/Desktop/Recent/LLDB-Debugger-Exploration/LLDB-Debugger-Exploration/ViewController.swift', Line = 60, exact_match = 0, locations = 1, resolved = 1, Hit count = 0 where = LLDB-Debugger-Exploration`LLDB_Debugger_Exploration.ViewController.viewDidLoad () -> () + 521 at ViewController.swift:60, address = 0x0000000109429609, resolved, hit count = 0 __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Unfortunately, you cannot pass parameters when executing a script file (unless you create an available variable in the script file).
The script subcommand provides advanced usage for managing custom Python scripts (add, delete, import, list). Script makes command automation possible. See the Python Scripting for LLDB guide for more information. In our example, we create a very simple script.py script that contains only the print_hello() command to print “Hello Debug!” To the console:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__import lldb
def print_hello(debugger, command, result, internal_dict):
print "Hello Debugger!"
def __lldb_init_module(debugger, internal_dict):
debugger.HandleCommand('command script add -f script.print_hello print_hello') // Handle script initialization and add command from this module
print 'The "print_hello" python command has been installed and is ready for use.' // Print confirmation that everything works
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
Next you need to import the Python script, which you can use directly with the script command:
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) command import ~/Desktop/script.py
The "print_hello" python command has been installed and is ready for use.
(lldb) print_hello
Hello Debugger!
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The status subcommand is used to quickly view the current platform information. Status prints out the SDK path, processor architecture, operating system version, and a list of available devices supported by the SDK.
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(lldb) platform status Platform: Ios-simulator Triple: x86_64-apple-macosx OS Version: 10.12.5 (16F73) Kernel: Darwin Kernel Version 16.6.0: Fri Apr 14 16:21:16 PDT 2017; Root :xnu-3789.60.24~6/RELEASE_X86_64 Hostname: 127.0.0.1 WorkingDir: / SDK Path: "/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator.sdk" Available devices: 614F8701-3D93-4B43-AE86-46A42FEB905A: iPhone 4s CD516CF7-2AE7-4127-92DF-F536FE56BA22: iPhone 5 0D76F30F-2332-4E0C-9F00-B86F009D59A3: iPhone 5s 3084003F-7626-462A-825B-193E6E5B9AA7: iPhone 6 ... __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
The LLDB visualization mode is not available in Xcode, but it can be used in terminals.
__Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__(LLDB) GUI __Fri Oct 13 2017 14:33:43 GMT+0800 (CST)____Fri Oct 13 2017 14:33:43 GMT+0800 (CST)__Copy the code
! This is how LLDB GUI mode looks like
conclusion
This article Outlines the power of the LLDB. Although the LLDB exists in our development environment, most people don’t realize its power. Hopefully this overview of the LLDB’s basic functionality and automated debugging has been helpful.
The article leaves out many features of LLDB. Visual debugging techniques are not mentioned. If you are interested in such topics, please leave them in the comments below. It’s nice to write something that people are interested in.
You are encouraged to open the terminal, open LLLDB, and type help. Very detailed documentation will be presented. Although this is a time-consuming process, you are expected to take the time to read the document. Only by mastering your tools can you become truly productive.
reference
- Official LLDB site — You’ll find here all possible materials related to LLDB. Documentation, guides, tutorials, sources and much more.
- LLDB Quick Start Guide by Apple — as usual, Apple has a great documentation. This guide will help you to get started with LLDB really quickly. Also, They’ve described how to do debugging with LLDB without Xcode
- How debuggers work: Part 1 — Basics — I enjoyed this series of articles a lot. It’s Just fantastic overview how debuggers really work. Article describes all underlying principles using code of hand-made debugger written in C. I strongly encourage you to read all parts of these great series (Part 2, Part 3).
- WWDC14 Advanced Swift Debugging in LLDB — great overview what’s new in LLDB in terms of Swift debugging. And how LLDB helps you be more productive with an overall debugging process using built-in functions and features.
- Introduction To LLDB Python Scripting — The Guide on Python Scripting for LLDB which allows you To start really quickly.
- Dancing in the Debugger. A Waltz with LLDB — A clever introduction to some LLDB basics. Some information is A bit outdated (like (lldb) thread return command, for example. Unfortunately, it doesn’t work with Swift properly because it can potentially bring some damage to reference counting). Still, It’s a great article to start your LLDB Journey.
Debugging Swift code with LLDB
Thanks qin Yun for the review of this article.
To contribute or translate InfoQ Chinese, please email [email protected]. You are also welcome to follow us on Sina Weibo (@InfoQ, @Ding Xiaoyun) and wechat (wechat id: InfoQChina).