-
Notifications
You must be signed in to change notification settings - Fork 3.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Autoware Transform Listener - performance improvement #5385
Comments
@yukkysaito could I ask you for your comments? Also I would appreciate if you could take a look into linked PR 🙏🏻 |
@amadeuszsz cc @mitsudome-r Thank you for your proposal. I think it’s an excellent suggestion, but I have three questions as a maintainer. First, you mentioned that the new feature, “Static Transform Listener,” has already been introduced in ROS 2 Rolling. Would it be possible to register this package in the autoware.repos so that it could also be used in Humble? Second, if we are able to utilize the Static Transform Listener in ROS 2 Rolling, what would be the advantages and disadvantages of using this feature? Would the proposed feature still offer better performance? Unless there is a specific reason, reinventing the wheel from a maintenance perspective isn’t ideal, as unique implementations can make Autoware more challenging for users. Third, if the response to the second question suggests that this proposal is indeed better, would it be possible to present this idea to the ROS 2 community? |
@yukkysaito thank you for answer!
This class resides in one of the common headers, therefore no extra package is needed. Putting
Rolling implementation requires explicitly defining type of listener. In other words, we need to define our listener as an instance of
If somehow we can use
We already use something custom in our nodes.
Good point. If we could apply changes directly into core ROS, I would make this implementation clearer by expanding tf2 API ( If we merge it, we could go for ROS 2 core changes. If it will become accepted and available in one of future distros, we remove our algorithm implementation and keep API. This way our class will expand base class by only different return type so it will act as a simple wrapper - no API changes, no difference from user point of view. |
I like this idea. If we are making changes to the use of tf in Autoware, I think we should consider feeding that back to ROS 2 upstream rather than keeping in our own code. Of course, we might have to keep the code in Autoware repository for a while until it becomes available, but I would like to have a plan to merge it to ROS 2. We can also consider backporting it to Humble/Jazzy once it is merged to rolling. |
Just as a minor feedback, we might want to also consider adding a parameter like "force_dynamic" in constructor for people who might want to use it like default TF Listener without changing the code to improve the performance for the first time. |
Thank you @mitsudome-r for your feedback.
This would be perfect. Attached PR consist of implementation which will be reduced significantly after merging this feature into core of ROS. Declarations in header of public methods will stay same - no next refactor needed.
We already have managed parameter with default value |
@amadeuszsz Thank you for your response. Overall, I think it looks good. If it can be implemented without dependency on Autoware, it might be better to manage the package in a separate repository and register it as a release repository. I will create a separate repository, so could you place the package there? |
Sure! Please, share the link. |
@amadeuszsz Sorry for the delay. I’ll create it today and send the link. The purpose is that separating the packages to be registered in the ROS 2 repository from those not to be registered makes the process easier. |
@amadeuszsz https://github.com/autowarefoundation/ManagedTransformListener |
Yes, but the class is called Maybe I provided some ambiguity using "managed listener" / "managed buffer" interchangeably in this thread, sorry. From code point of view this is buffer. |
@amadeuszsz Thanks |
@yukkysaito PR has been updated. Now ManagedTransformBuffer is located in external repository. Please, take a look on it 🙏🏻 |
Checklist
Description
Autoware runtime characterizes with hundreds nodes running at the same time. Each node uses Pub / Sub mechanism for communication. One of the most common subscribed topic is
/tf
and/tf_static
. These topics are subscribed implicitly while constructingTransformListener
object - after initialization a new nodecalled transform_listener_impl_xxxxxxxxxxxx
appears in nodes graph. For/tf
topic there is not much what could be done - ROS node needs an update for changing vehicle’s ego pose. For/tf_static
it seems there is no need for continuous subscription, but ROS 2 API does not allow for controlling listener behavior.In Autoware, most of nodes need a transformation between sensor kit links. By default approach we subscribe
/tf
topic even if it is not necessary. These extra nodes with high frequency/tf
topic subscription increase CPU utilization significantly.First, let’s clarify two definitions:
Static Transform Listener - a listener which reads requested transformations, save it to internal buffer, and stop subscribing topics. Next requests for same transformations just reads transformation from internal buffer.
Dynamic Transform Listener - a listener with default ROS 2 behavior, always reads recent transformations, regardless if it is
/tf
or/tf_static
.Recently ROS 2 Rolling got new feature which stands for Static Transform Listener. PR provides separate class for listener which subscribes only one topic -
/tf_static
. This solution potentially relaxes CPU load, but static TF listener has to be defined explicitly. Additionally, extra listener node will still remain in node graph. Nevertheless, this feature is not available in Humble or Jazzy distribution neither.Purpose
Reduce CPU load by getting rid of unnecessary TF listeners in ROS nodes graph.
Possible approaches
Current solution - explicitly defined type of listener
The solution is simple - it wrapping listener and buffer to a single class and explicitly declares if incoming transforms are static or dynamic. Unfortunately, we encountered and issue and all nodes which use this feature got has_static_tf_only parameter false by default. The next step would be populate to every single launch file and config file a parameter has_static_tf_only with appropriate value. This is possible, but based on number of Pilot.Auto repositories it will bring headaches. Secondly, exposing this parameter is likely to be bug-prone for the community and may sometimes lead to difficult to find sources of future issues.
Possible solution - implicitly defined type of listener
What if API does not expose any parameter and decide itself to behave like dynamic or static transform listener?
The diagram seems complicated, but this applies only for first TF requests. We can differentiate few behaviors for n_static occurrences of specific static TF and n_dynamic occurrences of specific dynamic TF:
n_static=1, n_dynamic=0 - initializes TF listeners, fills
static_tf_buffer
with resultant TF, reset TF listeners and returns TF.n_static>1, n_dynamic=0 - reads TF from
static_tf_buffer
and returns it.n_static>=0, n_dynamic=1 - initializes TF listeners, overrides listener function to dynamic listener (default ROS 2 approach), returns TF.
n_static>=0, n_dynamic>1 - behaves like a default ROS 2 TF listener.
First request might take slightly more time - it depends on request timeout (exposed as an argument in function call same way as it is done for lookupTransform). However, in general it will benefit with significant performance improvement during Autoware runtime. What is important here, if ROS node which use this class encounters dynamic TF, getTransform function will be overridden and starts behaving like a ROS-ish TF listener.
The CPU load comparison confirms performance improvement.
This improvement is caused by reduced number of TF listeners in nodes containers.
Old improved implementation by default does the difference only if
target_frame
=source_frame
. It can achieve same performance as new improved implementation, but this will require to sethas_static_tf_only
parameter to True.The proposed approach might fail in one scenario - static transformations change during the runtime. For now, Autoware does not consist of such a behavior. However, dynamic sensor calibration feature might introduce this problem. In that case, an extra service server for resetting
static_tf_buffer
could handle this.Another point is why we need
static_tf_buffer
at all and traversal function, which checks whether TF is static and fills this local buffer with that information. It could be integrated directly into casual ROS-ish TF buffer, but due to currentgeometry2
design, it would require big API changes. Without that changes and having current design in mind:we could expose call
tf_buffer_->wereStaticTFsOnlyRequested()
and turning off / ontf_listener_
easily. We would need just a TF tree placed somewhere in tf2 buffer - parent & child frame andis_static
information are already there. Nevertheless, it needs ROS 2 core changes and might be difficult to merge in short period of time.Definition of done
autoware_pointcloud_preprocessor
andautoware_ground_segmentation
)autoware_pointcloud_preprocessor
node containerThe text was updated successfully, but these errors were encountered: