This project contains Golang code, but it does not have the file hierarchy of
regular Golang projects (this is due to the use of Bazel as a build system).
The cmd/gofuse
utility enables to re-create the file hierarchy expected by Go
tools:
# Make sure to build to have all compile-time generated files
cd <path-to-agi-source>
bazel build pkg
# Prepare a agi-gofuse directory **outside of the AGI checkout directory**
mkdir <path-outside-agi-source>/agi-gofuse
# Run gofuse with the previous directory as a target
bazel run //cmd/gofuse -- -dir <path-to-agi-gofuse>
# Add agi-gofuse directory to your GOPATH environment variable.
# On Linux, with a bash shell, you can add the following to your ~/.bashrc file:
export GOPATH="${GOPATH:+${GOPATH}:}<path-to-agi-gofuse>"
# On other configurations, please search online how to add/edit environment variables.
If you encounter a symlink error on Windows like 'a required privilege is not held by the client', you have to use a command prompt with administrator privileges or enable Developer Mode as described here.
After adding the gofuse directory to your GOPATH, Go tools should work as expected. You can edit files under the newly populated gofuse directory. You should still compile under the original AGI checkout directory.
Despite its name, the gofuse command does NOT use FUSE (filesystem in userspace). It just creates directories and links to source files, including generated files. It is a good idea to re-run gofuse from time to time, to re-sync links to potential new files.
In terms of editor, VsCode has good Go support
thanks to its
Go extension.
With the GOPATH setup to gofuse and opening the <path-to-agi-gofuse>
directory,
as the root of your workspace, you should get some jump-to-definition and autocomplete
features working. Make sure to edit the files through their link found under the gofuse directory.
The recommended Golang debugger is delve. You can start a debug build of gapis or a client under this debugger.
To debug gapis, you can do:
dlv exec ./bazel-bin/pkg/gapis -- -enable-local-files -persist -rpc localhost:8888
You can then use dlv commands to add breakpoints, and actually start GAPIS, e.g.:
(dlv) break gapis/server/server.go:228
(dlv) continue # this actually starts gapis
See delve documentation on how to specify a breakpoint location, there are more convenient alternatives than
path/to/file:line
Once gapis is started, you can run a client to interact with it and hit somes breakpoints:
# in another terminal
./bazel-bin/pkg/gapit <verb> -gapis-port 8888 <verb args>
If you want to debug a client like gapit, just start it under dlv:
dlv exec ./bazel-bin/pkg/gapit <verb> <verb args>
To automate a delve startup sequence, you can edit a script of delve commands to be executed when delve starts. The script looks like:
# This is a comment.
break gapis/server/server.go:228
# add a second breakpoint, with a condition for it to trigger
break gapis/foo/bar.go:123
condition 2 some_variable == 42
# launch program
continue
And you can pass this script to delve using the --init
flag:
dlv exec --init my-delve-init-script.txt <program to debug...>
If you want to interact with the debugger via your editor or IDE, be aware that delve will think file paths start from the AGI top directory, and not your root directory. This is very likely due to Bazel compilation. You may have to find workarounds if you call delve from an editor/IDE which consider the file paths to start from another directory, typically your root directory. There may be a way to adjust using GOPATH to tell to your IDE a possible root for filename lookups.
See the workaround for VSCode below, any help to fix it for other IDEs is very welcome!
Follow these steps to use the delve debugger for Go with VSCode to debug gapis
.
-
Make sure to complete the Golang setup above for AGI.
-
Create a
launch.json
file under the workspace directory withCtrl + Shift + P
andDebug: Open launch.json
-
Paste the following as one of the launch configurations. This will ensure that there is a launch configuration for attaching to Delve.
{
...
"configurations": [
...,
{
"name": "Attach to Delve",
"type": "go",
"request": "attach",
"mode": "remote",
"apiVersion": 2,
"remotePath": "gapis/",
"cwd": "${workspaceFolder}/src/github.com/google/agi/gapis",
"dlvLoadConfig": {
"followPointers": true,
"maxVariableRecurse": 1,
"maxStringLen": 120,
"maxArrayValues": 120,
"maxStructFields": -1
},
"host": <host>,
"port": <port>,
},
],
...
}
As an example, <host>
could be 127.0.0.1
and <port>
could be 1234
.
- Start delve in headless mode in the AGI root folder.
dlv exec --headless --listen=<host>:<port> --api-version 2 ./bazel-bin/pkg/gapis -- <gapis-arguments>
The command below will allow using port 1234
(or any other preferred port) to connect to delve from VSCode.
dlv exec --headless --listen=127.0.0.1:1234 --api-version 2 ./bazel-bin/pkg/gapis -- -persist -rpc localhost:8888
-
Start debugging with
Debug->Start Debugging
(on Linux withF5
) and make sureAttach to Delve
is selected as the launch configuration. -
Now VSCode can interact with Delve and can be used for debugging
gapis
in VSCode UI instead of the command line. Enjoy your debugging :)
You can use the built-in logging functions to place debug prints.
In Golang:
import (
// ...
"github.com/google/gapid/core/log"
)
// ...
log.E(ctx, "Here debug print, myVar: %v", myVar)
In C++:
#include "core/cc/log.h"
// ...
GAPID_ERROR("Here debug print, myStr: %s", myStr)
The usual logging levels are available, listed for instance with gapit -fullhelp
:
$ ./bazel-bin/pkg/gapit -fullhelp
...
-log-level value
The severity to enable logs at [one of: "Verbose", "Debug", "Info", "Warning", "Error", "Fatal"] (default Info)
The Error level is recommended when adding debug print, to make sure it is not filtered away.