|
DBC Framework
DBC parsing and CAN signal decoding framework
|
This module provides a minimal but extensible DBC parser and runtime decode foundation for the FrameTap project.
Its purpose is to:
The implementation follows a simple and practical design:
This is not a full production-grade DBC implementation yet, but it is a strong architectural base.
The module is divided into two main paths.
Used for:
Model/ViewPipeline:
Used for:
Pipeline:
This separation is intentional.
The tree is useful for UI, but it is not the main data structure for runtime decoding.
A tree structure is convenient for browsing, but a runtime decoder needs something different:
Because of that, the design uses a dedicated runtime-ready structure:
This avoids forcing UI-oriented structures into a decode role they were not meant for.
Files:
These store a readable representation of the parsed DBC file.
Files:
These convert parsed DBC content into a tree hierarchy suitable for UI and model/view usage later.
Files:
These convert parsed DBC content into a structure optimized for decoding.
Files:
These perform actual decoding of raw CAN frames using DecodeDatabase.
Files:
These parse the DBC file itself.
File:
Used as a small integration example.
Represents one signal as parsed from the DBC file.
Fields:
namestartBitlengthisLittleEndianisSignedfactoroffsetminimummaximumunitreceiverscommentNotes:
receivers is a list because a signal may have more than one receiver ECUfactor and offset define physical conversionPhysical value rule:
Represents one frame as parsed from the DBC file.
Fields:
namecanIdisExtendedpgnhasPgndlctransmittercommentsignalsNotes:
signals is a list of SignalInfoisExtended is determined during CAN ID normalizationpgn is derived using simplified J1939 logic when applicableTop-level parsed DBC container.
Conceptually:
This is the central structure produced by DbcParser.
The UI tree contains three node types:
RootFrameSignalExample hierarchy:
Each node stores either:
The tree is intended for browsing and later Qt model integration.
It is not the primary runtime decode structure.
The decode layer exists so that decoding can be fast and independent from UI concerns.
Instead of searching a tree, the decoder uses a prepared database with direct lookup.
Runtime byte order enum:
IntelMotorolaThis is better for decode code than passing around raw DBC characters.
Numeric type enum:
UnsignedSignedThis is clearer than combining multiple boolean flags during runtime logic.
Represents one runtime-ready signal definition.
Fields:
namestartBitlengthbyteOrdervalueTypefactoroffsetminimummaximumunitreceiverscommentThis structure contains all information required for extracting and converting a signal value from raw frame data.
Represents one runtime-ready frame definition.
Fields:
namecanIdisExtendeddlcpgnhasPgntransmittercommentsignalsThis structure is used directly by the decoder.
Fast lookup key for runtime frame matching.
Fields:
canIdisExtendedThis matters because the same numeric identifier must not be confused between standard and extended frames.
Top-level runtime decode container.
Fields:
framesframeIndexByKeyConceptually:
This gives the decoder fast access to a frame definition using CAN ID and frame type.
Represents a raw CAN frame to decode.
Fields:
canIdisExtendeddataThis same structure can be used for:
Represents one decoded signal result.
Fields:
definitionrawValuephysicalValuevalidRepresents one decoded frame result.
Fields:
definitionsignalsvalidThis is the decoder output for one raw frame.
Main runtime decoder class.
Responsibilities:
Main methods:
FindFrame(...)Decode(...)The current parser supports the following DBC elements:
BO_SG_CM_ BO_CM_ SG_Example:
Parsed fields:
Example:
Parsed fields:
Frame comment example:
Signal comment example:
Stored in:
The parser normalizes frame identifiers.
Common DBC behavior:
The parser therefore stores:
canIdisExtended flagThis is important both for correct lookup and for future interoperability with live CAN APIs.
PGN is derived only when the frame is treated as extended.
The current logic is simplified J1939 extraction:
pfpsdpThis is enough for a practical start but should not be treated as full J1939 validation.
Typical runtime decode flow:
The decoder currently has separate extraction paths:
ExtractIntel(...)ExtractMotorola(...)This is important because byte order is not just metadata once decoding starts.
Intel and Motorola require different bit extraction logic.
This is one of the main reasons why the runtime decode layer should be explicit and prepared in advance.
A key design goal is that the same decoder should work for both:
That means this architecture supports:
This avoids duplicating decode logic in two separate parts of the application.
This module is meant to support at least the following FrameTap workflows:
Example combined workflow:
The tree exists for browsing.
However, runtime decode should not rely on tree traversal because that would introduce unnecessary coupling and inefficiency.
A runtime decoder needs:
That is why DecodeDatabase is a separate layer.
At the current stage, abstract factory is intentionally avoided.
The current design is already clean:
Introducing factory layers now would increase complexity without solving an immediate problem.
If later the project requires multiple output representations or multiple build strategies, that can be added then.
This is still a minimal implementation.
Not supported yet:
VAL_ tablesBA_ attributesBA_DEF_ definitionsMotorola extraction is implemented, but it should still be verified carefully against real-world DBC files and expected values.
A practical development order would be:
BO_SG_CM_isExtendedRecommended additions:
TreeNodeVAL_ support for enum-style signalsRecommended additions:
Advanced functionality:
The module does not depend on any specific build system.
It can be integrated with:
Just add the source files to the project.
This module is now split into two intentionally separate layers:
Used for:
Used for:
That separation is the main architectural improvement.
In short, the system now looks like this:
This gives FrameTap a much better foundation for real use, because both browsing and decoding are supported without forcing one representation to do the other's job.