-
Notifications
You must be signed in to change notification settings - Fork 1.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
Convert package to a pure CMake package #285
base: main
Are you sure you want to change the base?
Changes from 5 commits
3025939
380c4e4
c9da89d
61da1e2
40d2950
48f14ce
7b7f62b
2a76e1a
2528dcf
f60d49d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||
---|---|---|---|---|---|---|---|---|---|---|
@@ -1,31 +1,20 @@ | ||||||||||
cmake_minimum_required(VERSION 2.8.3) | ||||||||||
project(serial) | ||||||||||
cmake_minimum_required(VERSION 3.5.0) | ||||||||||
|
||||||||||
# Find catkin | ||||||||||
find_package(catkin REQUIRED) | ||||||||||
# General setup | ||||||||||
option(BUILD_SHARED_LIBS "Build using shared libraries" ON) | ||||||||||
moriarty marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||
|
||||||||||
set(PROJ_SOVERSION 1) | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why have this variable instead of just hardcoding this value a few lines down where it's used? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Similar to the recommendation to not use the One example where I've gone the other way from this recommendation though is in ros2_control with our use of the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Makes sense. I will remove the PROJ_SOVERSION and just use 1. all of the PROJECT_NAME variables come from cherry-pick from #231 and a later change in that PR makes it possible to have an alternative name for this library There are so many diverging forks of this library :'( But I agree and will change PROJECT_NAME back to serial for readability in a separate commit so if @wjwwood or @leamas chime in to this discussion it could always be reverted. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. PROJ_SOVERSION is actually crucial. It defines the binary compatibility level, and as such has been 1 so far. However, all serious packaging relies on a defined SOVERSION, without it all dependent packages will need to be recompiled for each and every change in the this lib, also the ones which are downwards compatible. EDIT: I actually explained that in the commit, see below There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That the library actually needs a new name is a reason, perhaps the only compelling one, to make a friendly fork. Forking is of course a bad thing, but perhaps the correct way to get out of the rabbit hole created by the too general serial/serial.h. It would make it easier to make big change while keeping the current lib as-is, for sure. Issues could be moved to a fork without too much hassle, either in the GUI one-by-one or on the command line using a bulk transfer. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @leamas Nobody is asking to stop specifying the SOVERSION. We're saying that we don't need an extra variable to hold the value There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Keeping the name is about traceability. Put it this way: if you replace SOVERSION with a an explicit constant like 1 or 2 it will need a comment explaining what it's all about. Such needs for comments is a sure sign of bad design Yes, cmake is a from time to time a hard language. However, this makes it even more important to stick to standard programming principles, one of which being to declare constants rather than using literal values. That is, SOVERSION is to be preferred to literal whatever. Would There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How is set_target_properties(${PROJECT_NAME} PROPERTIES
VERSION ${PROJECT_VERSION}
SOVERSION ${PROJ_SOVERSION}
PUBLIC_HEADER "${serial_HEADERS}"
) better than set_target_properties(${PROJECT_NAME} PROPERTIES
VERSION ${PROJECT_VERSION}
SOVERSION 1
PUBLIC_HEADER "${serial_HEADERS}"
) How is |
||||||||||
project(serial | ||||||||||
VERSION 1.2.1 | ||||||||||
DESCRIPTION "Cross-platform, Serial Port library written in C++" | ||||||||||
HOMEPAGE_URL "http://wjwwood.io/serial/" | ||||||||||
) | ||||||||||
moriarty marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||
|
||||||||||
if(APPLE) | ||||||||||
find_library(IOKIT_LIBRARY IOKit) | ||||||||||
find_library(FOUNDATION_LIBRARY Foundation) | ||||||||||
endif() | ||||||||||
|
||||||||||
if(UNIX AND NOT APPLE) | ||||||||||
# If Linux, add rt and pthread | ||||||||||
set(rt_LIBRARIES rt) | ||||||||||
set(pthread_LIBRARIES pthread) | ||||||||||
catkin_package( | ||||||||||
LIBRARIES ${PROJECT_NAME} | ||||||||||
INCLUDE_DIRS include | ||||||||||
DEPENDS rt pthread | ||||||||||
) | ||||||||||
else() | ||||||||||
# Otherwise normal call | ||||||||||
catkin_package( | ||||||||||
LIBRARIES ${PROJECT_NAME} | ||||||||||
INCLUDE_DIRS include | ||||||||||
) | ||||||||||
endif() | ||||||||||
|
||||||||||
## Sources | ||||||||||
set(serial_SRCS | ||||||||||
src/serial.cc | ||||||||||
|
@@ -47,7 +36,19 @@ else() | |||||||||
endif() | ||||||||||
|
||||||||||
## Add serial library | ||||||||||
set(serial_HEADERS | ||||||||||
include/serial/serial.h | ||||||||||
include/serial/v8stdint.h | ||||||||||
) | ||||||||||
Comment on lines
+37
to
+40
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
You don't need to specify headers as source files if you don't want to. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. v8stdint.h is actually problematic. It is now long time since I worked with this, but IIRC I think the situation is:
That is, the correct solution would be to use cmake to probe for On Fedora, v8stdint.h is part of the v8 package. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Of course, this is not only about including v8stdint.h or not in the package. It is also about conditionalizing the source files so they include the correct file, stdint.h or v8stdint.h. I made a quick hack for this in the Debian package, but it probably needs polish before being merged into the upstream package here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
This is IMHO the wrong way to go. Recommended cmake best practice is certainly to include all used files in the project. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I have seen a numbner of cmake projects, none of without the headers in the source lists. In this case, I think that you need to come up with some trustworthy source recommending not to include the headers. You have already started the list of functionality which breaks without a complete list of source files, including headers. I can assure you there is more, and that we currently not use it is not much of an argument. BTW: how would you install headers not being part of the sources? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
I'd argue the burden of proof is on the person who wants to write more code.
Nothing breaks for Visual Studio users. It's merely an inconvenience. Visual Studio users will also ask you use
Same way as always (prior to There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I simply don't have time for this, sorry. It's up to the maintainer to judge. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. My experience has been that adding headers is not required at all. It does help some editors who want to list "files used in targets" without involving a compiler, but that's all. Whether or not it's recommended by CMake, I'm not sure about. That being said, I don't care if they are included in this way. If it helps someone and they contribute it then I think it's fine. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Fair enough. But, as you said, cstdint.h is the better alternative to v8stdint.h making this to be serial.h only |
||||||||||
# Build, link and install main library | ||||||||||
add_library(${PROJECT_NAME} ${serial_SRCS}) | ||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
It aids in readability and There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 to this recommendation as it makes the cmake file much easier to read and understand. One of the hardest parts of cmake is reading the config files. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I agree here, but I'll wait for @wjwwood to chime in on this PR in general. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I am concerned this PR gets too big and goes the way of any of the other open PRs There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's fine with me, but it deviates from what it common in the ROS ecosystem. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Not only the ROS ecosystem, but the whole cmake community I would say. For that reason I think this could be left as-is There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @ChrisThrasher wrote No. This should IMHO be done like in e1dabd8, taking into account whether to include v8stdint.h or not. |
||||||||||
set_target_properties(${PROJECT_NAME} PROPERTIES | ||||||||||
VERSION ${PROJECT_VERSION} | ||||||||||
SOVERSION ${PROJ_SOVERSION} | ||||||||||
PUBLIC_HEADER "${serial_HEADERS}" | ||||||||||
) | ||||||||||
target_include_directories(${PROJECT_NAME} PUBLIC include) | ||||||||||
|
||||||||||
if(APPLE) | ||||||||||
target_link_libraries(${PROJECT_NAME} ${FOUNDATION_LIBRARY} ${IOKIT_LIBRARY}) | ||||||||||
elseif(UNIX) | ||||||||||
|
@@ -66,16 +67,27 @@ include_directories(include) | |||||||||
|
||||||||||
## Install executable | ||||||||||
install(TARGETS ${PROJECT_NAME} | ||||||||||
ARCHIVE DESTINATION ${CATKIN_PACKAGE_LIB_DESTINATION} | ||||||||||
LIBRARY DESTINATION ${CATKIN_PACKAGE_LIB_DESTINATION} | ||||||||||
RUNTIME DESTINATION ${CATKIN_GLOBAL_BIN_DESTINATION} | ||||||||||
ARCHIVE DESTINATION lib | ||||||||||
LIBRARY DESTINATION lib | ||||||||||
RUNTIME DESTINATION bin | ||||||||||
PUBLIC_HEADER DESTINATION include/${PROJECT_NAME} | ||||||||||
) | ||||||||||
|
||||||||||
## Install headers | ||||||||||
install(FILES include/serial/serial.h include/serial/v8stdint.h | ||||||||||
DESTINATION ${CATKIN_GLOBAL_INCLUDE_DESTINATION}/serial) | ||||||||||
DESTINATION include/serial) | ||||||||||
|
||||||||||
## Install CMake config | ||||||||||
install(FILES cmake/serialConfig.cmake | ||||||||||
DESTINATION share/serial/cmake) | ||||||||||
|
||||||||||
|
||||||||||
## Install package.xml | ||||||||||
install(FILES package.xml | ||||||||||
DESTINATION share/serial) | ||||||||||
Comment on lines
+80
to
+89
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This might be a good time to change to exported targets, and also consider putting the headers in a separate subfolder, which we've been doing in ROS 2 for a while now to promote better header isolation in a merged install space. I.e. we'd put the headers in |
||||||||||
|
||||||||||
## Tests | ||||||||||
if(CATKIN_ENABLE_TESTING) | ||||||||||
include(CTest) | ||||||||||
if(BUILD_TESTING) | ||||||||||
add_subdirectory(tests) | ||||||||||
endif() |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
get_filename_component(SERIAL_CMAKE_DIR "${CMAKE_CURRENT_LIST_FILE}" PATH) | ||
set(SERIAL_INCLUDE_DIRS "${SERIAL_CMAKE_DIR}/../../../include") | ||
find_library(SERIAL_LIBRARIES serial PATHS ${SERIAL_CMAKE_DIR}/../../../lib/serial) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If we're adding a config module, we should write a config module that exports a target instead of doing the old school approach of setting some variables that downstream projects have to use. That means we need to create an export set for the library target and install that export set then this file becomes a one liner that will look something like this: include(${CMAKE_CURRENT_LIST_DIR}/serial-targets.cmake) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @ChrisThrasher can you point to helpful documentation and examples on how to do this well? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm biased but this installation code I wrote I quite like. It's goes above and beyond to be as robust and idiomatic as possible. You don't need to copy everything it does but it should do a good job laying our the basic steps.
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
This is not old-school. The normal usage pattern would be something like below, assuming that the serial library lives in a subdirectory
This would just require a one-liner in CMakeLists.txt defining a ALIAS target. It relies on cmake's support for transitive dependencies. Adding such a target would simplify documentation and overall usage a lot. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, that's what I'm saying. We're in agreement. The variable-based approach is old school because targets are the modern way of doing things. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
This is actually a personal preference... I actually agree on the personal level, but let's face it: the use of ${PROJECT_NAME} is common enough in the cmake community to leave in place. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Don't mistake commonality for it being a good idea. Lots of bad practices are still popular but that should not be used as an argument in favor of perpetuating bad ideas. |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -19,6 +19,13 @@ | |
<author email="[email protected]">William Woodall</author> | ||
<author email="[email protected]">John Harrison</author> | ||
|
||
<buildtool_depend>catkin</buildtool_depend> | ||
<buildtool_depend>cmake</buildtool_depend> | ||
|
||
<!-- boost is only needed for test/proof_of_concepts which are not enabled by default --> | ||
<!--test_depend>boost</test_depend--> | ||
<test_depend>gtest</test_depend> | ||
|
||
<export> | ||
<build_type>cmake</build_type> | ||
</export> | ||
</package> |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,12 +1,19 @@ | ||
if(UNIX) | ||
catkin_add_gtest(${PROJECT_NAME}-test unix_serial_tests.cc) | ||
target_link_libraries(${PROJECT_NAME}-test ${PROJECT_NAME}) | ||
find_package(GTest REQUIRED) | ||
include_directories(${GTEST_INCLUDE_DIRS}) | ||
moriarty marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
add_executable(${PROJECT_NAME}-test unix_serial_tests.cc) | ||
target_link_libraries(${PROJECT_NAME}-test ${PROJECT_NAME} ${GTEST_LIBRARIES}) | ||
if(NOT APPLE) | ||
target_link_libraries(${PROJECT_NAME}-test util) | ||
endif() | ||
add_test("${PROJECT_NAME}-test-gtest" ${PROJECT_NAME}-test) | ||
|
||
if(NOT APPLE) # these tests are unreliable on macOS | ||
catkin_add_gtest(${PROJECT_NAME}-test-timer unit/unix_timer_tests.cc) | ||
target_link_libraries(${PROJECT_NAME}-test-timer ${PROJECT_NAME}) | ||
add_executable(${PROJECT_NAME}-test-timer unit/unix_timer_tests.cc) | ||
target_link_libraries(${PROJECT_NAME}-test-timer ${PROJECT_NAME} ${GTEST_LIBRARIES}) | ||
add_test("${PROJECT_NAME}-test-timer-gtest" ${PROJECT_NAME}-test-timer) | ||
endif() | ||
# Boost is only required for tests/proof_of_concepts which are not enabled by default | ||
# find_package(Boost REQUIRED) | ||
endif() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why such an old version? I'm not sure what non-EOL platforms these days ship anything older than 3.16.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
RHEL8 ships with 3.16, the oldest version of CMake in a distro currently supported by a tier 1 or 2 versions of ROS to give you more context. @ChrisThrasher, do you mind helping explain the benefits of using a newer version of CMake and provide some references in case @moriarty would like more information about this recommendation?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The main benefit is that it means you opt-in to many new and improved default behaviors (what CMake calls "policies") that have been added between version 3.5 and 3.16.
https://cmake.org/cmake/help/latest/manual/cmake-policies.7.html
On top of that, nobody is actually using 3.5 or testing with 3.5 so while we can claim this script doesn't use features added after 3.5 we can't be sure that unless we add a CI job that uses 3.5. CMake will let you use newer features even if your minimum version is very old.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I took 3.5 because I got a CMake warning that said something and suggested using at least 3.5
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also when I saw that warning, I just cherr-picked because it was on the other open PR which is what is being released into debian as
libcxx-serial-dev
: c9da89d but when I tried to use that package I had to add some define around the includes of serial (WIP testing here: PickNikRobotics/ros2_robotiq_gripper#22)There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
7b7f62b I've set this to 3.16