5.2 KiB
DebugServer
The DebugServer component exposes an interface to the connected target, for third-party programs such as IDEs. The
DebugServer runs on a dedicated thread. The entry point is DebugServerComponent::run(). Bloom's main thread will start
the DebugServer thread once the TargetController has been started. See Applciation::startDebugServer() for more.
The DebugServer is designed to accommodate numerous server implementations. The interface exposed by the server is
implementation-defined. For example, the AVR GDB server exposes an interface to the
connected AVR target, by implementing the
GDB Remote Serial Protocol, over a TCP/IP connection.
Each server must implement the interface defined in the ServerInterface class.
At start up, the DebugServer will select the appropriate server implementation, based on the user's project
configuration (bloom.yaml - see DebugServerConfig). Each server implementation is mapped to a
config name, which is to be used in the project configuration file. For the mapping, see
DebugServerComponent::getAvailableServersByName(). After initialising the server (via ServerInterface::init()),
control of the DebugServer thread will then be handed over to the server implementation (via ServerInterface::run()).
For more on this, see DebugServerComponent::startup() and DebugServerComponent::run().
Servicing events
During star tup, the DebugServer will register event handlers for certain events. Once control of the DebugServer thread
has been handed over to the selected server implementation, the server must ensure that any incoming events are
processed ASAP. How this is done is implementation-defined. A reference to the DebugServer's event listener
(DebugServerComponent::eventListener) can be passed to the server if need be (see
DebugServerComponent::getAvailableServersByName() for an example of this).
A server implementation does not have to service events - it can simply return from ServerInterface::run() upon
an event being triggered. Returning from that function will allow control of the thread to be handed back to
DebugServerComponent::run(), where any pending events will be processed. Afterwards, if the DebugServer is still
running (it hasn't received an event which triggers a shutdown), the ServerInterface::run() function will be called
again, and control will be handed back to the server implementation. The AVR GDB server implementation employs this
method to ensure that events are processed ASAP. See the relevant documentation for more.
Any blocking I/O employed by a server implementation can support event interrupts using an
EventFdNotifier and EpollInstance.
Key points:
- The
EventFdNotifierclass is an RAII wrapper for a Linux eventfd object. It implements theNotifierInterface. An event can be recorded against the eventfd object via a call toEventFdNotifier::notify(). - The
EventListenerclass can accept aNotifierInterfaceobject viaEventListener::setInterruptEventNotifier(). If aNotifierInterfacehas been set on anEventListener, theEventListenerwill callNotifierInterface::notify()everytime an event is registered for that listener. - The
EpollInstanceclass is an RAII wrapper for a Linux epoll instance. It allows us to wait for any activity on a set of file descriptors. File descriptors can be added and removed from the epoll instance viaEpollInstance::addEntry()andEpollInstance::removeEntry(). CallingEpollInstance::waitForEvent()will block until there is activity on at least one of the file descriptors, or a timeout has been reached.
With an EventFdNotifier and EpollInstance, one can perform a blocking I/O operation which can be interrupted by an
event. For an example of this, see the AVR GDB server implementation - it employs the method described above to allow
the interruption of blocking I/O operations when an event is triggered. Specifically,
GdbRspDebugServer::waitForConnection() or
Gdb::Connection::read().
Server implementations
Currently, there is only one server implementation. Others may be added upon request.
| Server Name | Brief Description | Documentation |
|---|---|---|
| AVR GDB Server | An AVR-specific implementation of the GDB Remote Serial Protocol over TCP/IP. | /src/DebugServer/Gdb/README.md |
Adding new server implementations
If you're considering adding a new server implementation, please discuss this with me (Nav) before you begin. Creating a new server implementation is not a trivial task.