Skip to content
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

[opentitan] Integrate OpenTitan Peripherals #11

Draft
wants to merge 19 commits into
base: devel
Choose a base branch
from

Conversation

viv-eth
Copy link
Contributor

@viv-eth viv-eth commented Nov 29, 2024

This PR introduces support for OpenTitan peripherals and adjusts the build system to improve integration and build consistency.

OpenTitan Integration and Build System Improvements

  1. Integration of OpenTitan Peripherals
    • OpenTitan is now added as an external project using sparse checkout. This method ensures that only the necessary files are pulled from the repository, reducing overhead.
    • When the repository already exists locally, adding new sources to the SPARSE_PATTERNS will only fetch the additional files, avoiding a full repository download. Take care to ensure everything is up to date.
    • ⚠️ Warning: When adding new sources, ensure OPENTITAN_SOURCES and SPARSE_PATTERNS are updated accordingly.
  2. Compilation and Linking
    • The pulled sources are compiled into a public library and linked against the runtime.

@viv-eth viv-eth mentioned this pull request Dec 2, 2024
@viv-eth viv-eth requested review from Scheremo and Xeratec and removed request for Scheremo December 2, 2024 15:54
@viv-eth viv-eth marked this pull request as draft December 3, 2024 16:05
Copy link
Collaborator

@Scheremo Scheremo left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Left some comments - quite a bit in opentitan.cmake doesn't intuitively make sense to me, maybe you can elaborate. IMO what you're doing in there is a mix of installation / setup and build flow. I would take this apart and only keep the build flow in the build flow.

CMakeLists.txt Show resolved Hide resolved
CMakeLists.txt Outdated Show resolved Hide resolved
Comment on lines 32 to 97
ExternalProject_Add(
opentitan
DOWNLOAD_COMMAND
git clone --no-checkout https://github.com/lowRISC/opentitan.git ${OPENTITAN_DIR}
COMMAND git -C ${OPENTITAN_DIR} config core.sparseCheckout true
COMMAND ${CMAKE_COMMAND} -E copy ${SPARSE_CHECKOUT_FILE} ${OPENTITAN_DIR}/.git/info/sparse-checkout
COMMAND git -C ${OPENTITAN_DIR} checkout master
UPDATE_COMMAND "" # Disable updates to avoid unexpected changes
CONFIGURE_COMMAND "" # No configure step needed
BUILD_COMMAND "" # No build step needed
INSTALL_COMMAND "" # No install step needed
SOURCE_DIR ${OPENTITAN_DIR}
)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd strongly suggest picking a commit hash / version number that corresponds to the RTL groundtruth. Most likely should be set in targets.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It makes perfect sense; I am now cloning it based on a target-specific commit hash. C.f. b0f9d16.

Comment on lines 11 to 33
if(EXISTS ${OPENTITAN_DIR})
file(REMOVE_RECURSE ${OPENTITAN_DIR})
endif()

set(SPARSE_CHECKOUT_FILE ${CMAKE_CURRENT_BINARY_DIR}/sparse-checkout)

set(SPARSE_PATTERNS
"util/*"
"hw/ip/*"
"sw/device/lib/base/*"
"sw/device/lib/base/internal/*"
"sw/device/lib/dif/*"
"sw/device/lib/dif/autogen/*"
)

file(WRITE ${SPARSE_CHECKOUT_FILE} "")
foreach(PATTERN ${SPARSE_PATTERNS})
file(APPEND ${SPARSE_CHECKOUT_FILE} "${PATTERN}\n")
endforeach()
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do this?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

View comment below.

Comment on lines 56 to 124
add_custom_command(
OUTPUT ${UART_REGS_H}
COMMAND ${Python3_EXECUTABLE} ${REGTOOL_PY} ${UART_HJSON} -D > ${UART_REGS_H}
DEPENDS ${REGTOOL_PY} ${UART_HJSON}
COMMENT "Generating uart_regs.h"
)

add_custom_target(
generate_uart_regs ALL
DEPENDS ${UART_REGS_H}
)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we externalize this? I would do this in a setup procedure, not the main build flow.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the feedback! I see the concern about keeping the uart_regs.h generation within the main build flow, and I’d like to clarify why this step is currently integrated and how externalizing it might affect the workflow.

The uart_regs.h is necessary for building the opentitan_lib. Without it, the library cannot be compiled. Externalizing this process entirely (e.g., as a manual pre-build step) would introduce additional steps for developers, like running a separate script before invoking the main build. Or maybe I am missing the point of your comment 😅

If you’re suggesting a more decoupled approach (e.g., fully externalizing the process to a pre-build script or tool), let me know what you envision.

  • Would you prefer having a dedicated setup.sh or similar to prepare the repository before building?
  • Or is the suggestion to make the build system more modular while retaining the automation?

Let me know what you think @Scheremo .

Copy link
Collaborator

@Scheremo Scheremo Dec 19, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Seems like a perfect use case for a makefile, or an installation step in your cmake flow imo. Redownloading the entire repo and building everything from scratch requires an internet connection, but it's not really necessary. I'd add a very simple setup step (note: makefile, not shell script) if necessary to generate these headers. Another fairly reasonable option is to check in generated headers in a patch.

Another gripe of mine is that CMake is not meant to run python scripts, and having it do that makes it much harder to understand what's going on. Debugging my C build flow should not entail debugging Python. The installation process, however, is a different story :)

Comment on lines 68 to 80
add_dependencies(generate_uart_regs opentitan)

ExternalProject_Get_Property(opentitan SOURCE_DIR)

file(GLOB_RECURSE OPENTITAN_SOURCES
"${OPENTITAN_SW_DIR}/device/lib/base/*.c"
"${OPENTITAN_SW_DIR}/device/lib/dif/*.c"
"${OPENTITAN_SW_DIR}/device/lib/dif/autogen/*.c"
"${OPENTITAN_SW_DIR}/device/lib/runtime/*.c"
)

list(FILTER OPENTITAN_SOURCES EXCLUDE REGEX ".*crc32.*")
list(FILTER OPENTITAN_SOURCES EXCLUDE REGEX ".*hardened.*")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is some brute-forcing going on here. What's up with crc32 and hardened?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I’ve noticed that the opentitan repository includes a considerable number of files that are either unnecessary or unsupported on our platforms. As you might recall from my discussion with Pirmin from OpenTitan, he suggested using the OT_PLATFORM_RV32 flag for 32-bit RISC-V platforms. However, this flag also enables crc32 instructions (cyclic redundancy check), which are part of the Bitmanip extensions that our toolchain does not support.

Given this, I don’t see the benefit of including the entire repository at this stage. Doing so introduces some drawbacks, such as:

  1. Slowing down the cloning process and the building of opentitan_lib.
  2. Including files with instructions that we currently cannot support.

For now, I believe it makes sense to keep the repository to a minimal set of essential files. As we progress, we can always revisit and add more files if the need arises.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Definitely a discussion worthy of a comment - if I see these stray lines of filtering I start to wonder :)

Point 1, however, should be mute, as built files should not be rebuilt (read: I would set up an installation and reuse things that can be reused).
Point 2 is fair enough. Adding it here is a decent choice imo, and I am a bit dubious whether this approach will scale to targets with different peripheral configurations - the drivers to be used are target-specific after all.

Comment on lines 98 to 150
target_link_libraries(opentitan_lib
PUBLIC
runtime
)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not sure this is not the other way around... What's the thought process?

Copy link
Contributor Author

@viv-eth viv-eth Dec 19, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Indeed! This is addressed in a2974ab.

@Xeratec Xeratec added the enhancement New feature or request label Dec 17, 2024
@viv-eth viv-eth force-pushed the vivianep/opentitan-peripherals branch from d7f9aab to 2c298d9 Compare December 19, 2024 11:00
Copy link
Collaborator

@Xeratec Xeratec left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good in principle but I suggest a different folder structure. Additionally, unfortunately, the build flow only works for the chimera-open target as the opentitan.cmake is only included there, but drivers/CMakeLists.txt runtime_host is always linked against opentitan_lib. My suggestions should also fix this issue.

I suggest a more centralized folder structure where you put the opentitan.cmake inside drivers/opentitan, rename it to CMakeLists.txt.
Then the only thing a target needs to do is to add_subfolder(opentitan) (handled via the DRIVER_MAPPINGS). In this case, you then can clone the external source files into drivers/opentitan/third_party).

I also just saw that the cloned files are not included in the .gitignore. Please make sure that no external files will be accidentally check in.

cmake/opentitan.cmake Outdated Show resolved Hide resolved
drivers/CMakeLists.txt Outdated Show resolved Hide resolved
cmake/opentitan.cmake Show resolved Hide resolved
targets/chimera-open/CMakeLists.txt Outdated Show resolved Hide resolved
@viv-eth
Copy link
Contributor Author

viv-eth commented Dec 20, 2024

For every driver, include all necessary opentitan sources into the sub-folder of the driver, e.g. drivers/uart_opentitan/thirdparty (all files there are not maintained by us). Remove drivers/opentitan and add CMakeLists.txt on driver top-level. Add uart driver with HAL as an example.

Copy link
Collaborator

@Xeratec Xeratec left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks already very good! One minor comment about the driver inclusion, and I also noticed that the build flow now has some Python packages imposed by OpenTitan. They are listed in https://github.com/lowRISC/opentitan/blob/master/pyproject.toml or https://github.com/lowRISC/opentitan/blob/master/pyproject.toml. Please make sure to handle/explain how to set up and install these dependencies.

Comment on lines 17 to 18
# VIVIANEP: Choose driver versions (if applicable)
set(UART_DRIVER_VERSION "opentitan" CACHE STRING "UART driver version")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why this additional abstraction and not just include uart_opentitan above? I don't see any benefit but potential unclarity.

@@ -8,7 +8,7 @@
# Define mappings for drivers
set(DRIVER_MAPPINGS
chimera-convolve:cluster
chimera-open:cluster
chimera-open:cluster,uart_${UART_DRIVER_VERSION}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please see the comment below. I am not a big fan of the uart_${UART_DRIVER_VERSION} and suggest uart_opentitan.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants