-
Notifications
You must be signed in to change notification settings - Fork 569
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
examples: New app to build Rust with Cargo
Build Rust applictions with cargo is the most commn way, and it's more easy to cooporate with Rust ecosystem. This example shows how to use cargo to build a simple hello world application. And please notice that you need to install nighly version of rustc to support this feature, any version after rust-lang/rust#127755 is merged, can use NuttX as cargo target directly. Build ----- To build hello_rust_cargo application, you can use any target that based on RISCV32IMAC, for example: ``` cmake -B build -DBOARD_CONFIG=rv-virt:nsh -GNinja . ``` And disable ARCH_FPU in menuconfig, since the hard coded target triple in this demo is `riscv32imac`. Signed-off-by: Huang Qi <[email protected]>
- Loading branch information
Showing
10 changed files
with
493 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,166 @@ | ||
# ############################################################################## | ||
# cmake/nuttx_add_rust.cmake | ||
# | ||
# Licensed to the Apache Software Foundation (ASF) under one or more contributor | ||
# license agreements. See the NOTICE file distributed with this work for | ||
# additional information regarding copyright ownership. The ASF licenses this | ||
# file to you under the Apache License, Version 2.0 (the "License"); you may not | ||
# use this file except in compliance with the License. You may obtain a copy of | ||
# the License at | ||
# | ||
# http://www.apache.org/licenses/LICENSE-2.0 | ||
# | ||
# Unless required by applicable law or agreed to in writing, software | ||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||
# License for the specific language governing permissions and limitations under | ||
# the License. | ||
# | ||
# ############################################################################## | ||
|
||
include(nuttx_parse_function_args) | ||
|
||
# ~~~ | ||
# Convert architecture type to Rust NuttX target | ||
# | ||
# Supported architectures: | ||
# - armv7a: armv7a-nuttx-eabi, armv7a-nuttx-eabihf | ||
# - thumbv6m: thumbv6m-nuttx-eabi | ||
# - thumbv7a: thumbv7a-nuttx-eabi, thumbv7a-nuttx-eabihf | ||
# - thumbv7m: thumbv7m-nuttx-eabi | ||
# - thumbv7em: thumbv7em-nuttx-eabihf | ||
# - thumbv8m.main: thumbv8m.main-nuttx-eabi, thumbv8m.main-nuttx-eabihf | ||
# - thumbv8m.base: thumbv8m.base-nuttx-eabi, thumbv8m.base-nuttx-eabihf | ||
# - riscv32: riscv32imc/imac/imafc-unknown-nuttx-elf | ||
# - riscv64: riscv64imac/imafdc-unknown-nuttx-elf | ||
# | ||
# Inputs: | ||
# ARCHTYPE - Architecture type (e.g. thumbv7m, riscv32) | ||
# ABITYPE - ABI type (e.g. eabi, eabihf) | ||
# CPUTYPE - CPU type (e.g. cortex-m4, sifive-e20) | ||
# | ||
# Output: | ||
# OUTPUT - Rust target triple (e.g. riscv32imac-unknown-nuttx-elf, | ||
# thumbv7m-nuttx-eabi, thumbv7em-nuttx-eabihf) | ||
# ~~~ | ||
|
||
function(nuttx_rust_target_triple ARCHTYPE ABITYPE CPUTYPE OUTPUT) | ||
if(ARCHTYPE MATCHES "thumb") | ||
if(ARCHTYPE MATCHES "thumbv8m") | ||
# Extract just the base architecture type (thumbv8m.main or thumbv8m.base) | ||
if(ARCHTYPE MATCHES "thumbv8m.main") | ||
set(ARCH_BASE "thumbv8m.main") | ||
elseif(ARCHTYPE MATCHES "thumbv8m.base") | ||
set(ARCH_BASE "thumbv8m.base") | ||
else() | ||
# Otherwise determine if we should use thumbv8m.main or thumbv8m.base | ||
# based on CPU type | ||
if(CPUTYPE MATCHES "cortex-m23") | ||
set(ARCH_BASE "thumbv8m.base") | ||
else() | ||
set(ARCH_BASE "thumbv8m.main") | ||
endif() | ||
endif() | ||
set(TARGET_TRIPLE "${ARCH_BASE}-nuttx-${ABITYPE}") | ||
else() | ||
set(TARGET_TRIPLE "${ARCHTYPE}-nuttx-${ABITYPE}") | ||
endif() | ||
elseif(ARCHTYPE STREQUAL "riscv32") | ||
if(CPUTYPE STREQUAL "sifive-e20") | ||
set(TARGET_TRIPLE "riscv32imc-unknown-nuttx-elf") | ||
elseif(CPUTYPE STREQUAL "sifive-e31") | ||
set(TARGET_TRIPLE "riscv32imac-unknown-nuttx-elf") | ||
elseif(CPUTYPE STREQUAL "sifive-e76") | ||
set(TARGET_TRIPLE "riscv32imafc-unknown-nuttx-elf") | ||
else() | ||
set(TARGET_TRIPLE "riscv32imc-unknown-nuttx-elf") | ||
endif() | ||
elseif(ARCHTYPE STREQUAL "riscv64") | ||
if(CPUTYPE STREQUAL "sifive-s51") | ||
set(TARGET_TRIPLE "riscv64imac-unknown-nuttx-elf") | ||
elseif(CPUTYPE STREQUAL "sifive-u54") | ||
set(TARGET_TRIPLE "riscv64imafdc-unknown-nuttx-elf") | ||
else() | ||
set(TARGET_TRIPLE "riscv64imac-unknown-nuttx-elf") | ||
endif() | ||
endif() | ||
set(${OUTPUT} | ||
${TARGET_TRIPLE} | ||
PARENT_SCOPE) | ||
endfunction() | ||
|
||
# ~~~ | ||
# nuttx_add_rust | ||
# | ||
# Description: | ||
# Build a Rust crate and add it as a static library to the NuttX build system | ||
# | ||
# Example: | ||
# nuttx_add_rust( | ||
# CRATE_NAME | ||
# hello | ||
# CRATE_PATH | ||
# ${CMAKE_CURRENT_SOURCE_DIR}/hello | ||
# ) | ||
# ~~~ | ||
|
||
function(nuttx_add_rust) | ||
|
||
# parse arguments into variables | ||
nuttx_parse_function_args( | ||
FUNC | ||
nuttx_add_rust | ||
ONE_VALUE | ||
CRATE_NAME | ||
CRATE_PATH | ||
REQUIRED | ||
CRATE_NAME | ||
CRATE_PATH | ||
ARGN | ||
${ARGN}) | ||
|
||
# Determine build profile based on CONFIG_DEBUG_FULLOPT | ||
if(CONFIG_DEBUG_FULLOPT) | ||
set(RUST_PROFILE "release") | ||
else() | ||
set(RUST_PROFILE "debug") | ||
endif() | ||
|
||
# Get the Rust target triple | ||
message(STATUS "LLVM_ARCHTYPE: ${LLVM_ARCHTYPE}") | ||
nuttx_rust_target_triple(${LLVM_ARCHTYPE} ${LLVM_ABITYPE} ${LLVM_CPUTYPE} | ||
RUST_TARGET) | ||
|
||
# Set up build directory in current binary dir | ||
set(RUST_BUILD_DIR ${CMAKE_CURRENT_BINARY_DIR}/${CRATE_NAME}) | ||
set(RUST_LIB_PATH | ||
${RUST_BUILD_DIR}/${RUST_TARGET}/${RUST_PROFILE}/lib${CRATE_NAME}.a) | ||
|
||
# Create build directory | ||
file(MAKE_DIRECTORY ${RUST_BUILD_DIR}) | ||
|
||
# Add a custom command to build the Rust crate | ||
add_custom_command( | ||
OUTPUT ${RUST_LIB_PATH} | ||
COMMAND | ||
cargo build --${RUST_PROFILE} -Zbuild-std=std,panic_abort --manifest-path | ||
${CRATE_PATH}/Cargo.toml --target ${RUST_TARGET} --target-dir | ||
${RUST_BUILD_DIR} | ||
COMMENT "Building Rust crate ${CRATE_NAME}" | ||
VERBATIM) | ||
|
||
# Add a custom target that depends on the built library | ||
add_custom_target(${CRATE_NAME}_build ALL DEPENDS ${RUST_LIB_PATH}) | ||
|
||
# Add imported library target | ||
add_library(${CRATE_NAME} STATIC IMPORTED GLOBAL) | ||
set_target_properties(${CRATE_NAME} PROPERTIES IMPORTED_LOCATION | ||
${RUST_LIB_PATH}) | ||
|
||
# Add the Rust library to NuttX build | ||
nuttx_add_extra_library(${RUST_LIB_PATH}) | ||
|
||
# Ensure the Rust library is built before linking | ||
add_dependencies(${CRATE_NAME} ${CRATE_NAME}_build) | ||
|
||
endfunction() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
# ############################################################################## | ||
# apps/examples/hello_rust_cargo/CMakeLists.txt | ||
# | ||
# Licensed to the Apache Software Foundation (ASF) under one or more contributor | ||
# license agreements. See the NOTICE file distributed with this work for | ||
# additional information regarding copyright ownership. The ASF licenses this | ||
# file to you under the Apache License, Version 2.0 (the "License"); you may not | ||
# use this file except in compliance with the License. You may obtain a copy of | ||
# the License at | ||
# | ||
# http://www.apache.org/licenses/LICENSE-2.0 | ||
# | ||
# Unless required by applicable law or agreed to in writing, software | ||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||
# License for the specific language governing permissions and limitations under | ||
# the License. | ||
# | ||
# ############################################################################## | ||
|
||
if(CONFIG_EXAMPLES_HELLO_RUST_CARGO) | ||
|
||
# Build the Rust crate using nuttx_add_rust | ||
nuttx_add_rust(CRATE_NAME hello CRATE_PATH ${CMAKE_CURRENT_SOURCE_DIR}/hello) | ||
|
||
nuttx_add_application( | ||
NAME ${CONFIG_EXAMPLES_HELLO_RUST_CARGO_PROGNAME} STACKSIZE | ||
${CONFIG_EXAMPLES_HELLO_STACKSIZE} PRIORITY | ||
${CONFIG_EXAMPLES_HELLO_PRIORITY}) | ||
|
||
endif() # CONFIG_EXAMPLES_HELLO_RUST_CARGO |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
# | ||
# For a description of the syntax of this configuration file, | ||
# see the file kconfig-language.txt in the NuttX tools repository. | ||
# | ||
|
||
config EXAMPLES_HELLO_RUST_CARGO | ||
tristate "\"Hello, Rust!\" example with Cargo" | ||
default n | ||
---help--- | ||
Enable the \"Hello, Rust!\" example using Cargo to build. | ||
|
||
if EXAMPLES_HELLO_RUST_CARGO | ||
|
||
config EXAMPLES_HELLO_RUST_CARGO_PROGNAME | ||
string "Program name" | ||
default "hello_rust_cargo" | ||
---help--- | ||
This is the name of the program that will be used when the | ||
program is installed. | ||
|
||
config EXAMPLES_HELLO_RUST_CARGO_PRIORITY | ||
int "Hello Rust task priority" | ||
default 100 | ||
|
||
config EXAMPLES_HELLO_RUST_CARGO_STACKSIZE | ||
int "Hello Rust stack size" | ||
default DEFAULT_TASK_STACKSIZE | ||
|
||
endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
############################################################################ | ||
# apps/examples/hello_rust_cargo/Make.defs | ||
# | ||
# Licensed to the Apache Software Foundation (ASF) under one or more | ||
# contributor license agreements. See the NOTICE file distributed with | ||
# this work for additional information regarding copyright ownership. The | ||
# ASF licenses this file to you under the Apache License, Version 2.0 (the | ||
# "License"); you may not use this file except in compliance with the | ||
# License. You may obtain a copy of the License at | ||
# | ||
# http://www.apache.org/licenses/LICENSE-2.0 | ||
# | ||
# Unless required by applicable law or agreed to in writing, software | ||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||
# License for the specific language governing permissions and limitations | ||
# under the License. | ||
# | ||
############################################################################ | ||
|
||
include $(APPDIR)/tools/Rust.mk | ||
|
||
ifneq ($(CONFIG_EXAMPLES_HELLO_RUST_CARGO),) | ||
CONFIGURED_APPS += $(APPDIR)/examples/hello_rust_cargo | ||
EXTRA_LIBS += $(call RUST_GET_BINDIR,hello,$(APPDIR)/examples/hello_rust_cargo) | ||
endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
############################################################################ | ||
# apps/examples/hello_rust/Makefile | ||
# | ||
# Licensed to the Apache Software Foundation (ASF) under one or more | ||
# contributor license agreements. See the NOTICE file distributed with | ||
# this work for additional information regarding copyright ownership. The | ||
# ASF licenses this file to you under the Apache License, Version 2.0 (the | ||
# "License"); you may not use this file except in compliance with the | ||
# License. You may obtain a copy of the License at | ||
# | ||
# http://www.apache.org/licenses/LICENSE-2.0 | ||
# | ||
# Unless required by applicable law or agreed to in writing, software | ||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||
# License for the specific language governing permissions and limitations | ||
# under the License. | ||
# | ||
############################################################################ | ||
|
||
include $(APPDIR)/Make.defs | ||
|
||
# Hello, Rust! built-in application info | ||
|
||
PROGNAME = $(CONFIG_EXAMPLES_HELLO_RUST_CARGO_PROGNAME) | ||
PRIORITY = $(CONFIG_EXAMPLES_HELLO_RUST_CARGO_PRIORITY) | ||
STACKSIZE = $(CONFIG_EXAMPLES_HELLO_RUST_CARGO_STACKSIZE) | ||
MODULE = $(CONFIG_EXAMPLES_HELLO_RUST_CARGO) | ||
|
||
context:: | ||
$(call RUST_CARGO_BUILD,hello,$(APPDIR)/examples/hello_rust_cargo) | ||
|
||
clean:: | ||
$(call RUST_CARGO_CLEAN,hello,$(APPDIR)/examples/hello_rust_cargo) | ||
|
||
include $(APPDIR)/Application.mk |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
target |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
[package] | ||
name = "hello" | ||
version = "0.1.0" | ||
edition = "2021" | ||
|
||
[lib] | ||
crate-type = ["staticlib"] | ||
|
||
[profile.dev] | ||
panic = "abort" | ||
|
||
# Special hanlding for the panic! macro, can be removed once | ||
# the libstd port for NuttX is complete. | ||
[profile.release] | ||
panic = "abort" | ||
lto = true | ||
|
||
[dependencies] | ||
serde = { version = "1.0", features = ["derive"] } | ||
serde_json = "1.0" | ||
|
||
tokio = { version = "1", features = ["rt"] } |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,56 @@ | ||
extern crate serde; | ||
extern crate serde_json; | ||
|
||
use serde::{Deserialize, Serialize}; | ||
|
||
#[derive(Serialize, Deserialize)] | ||
struct Person { | ||
name: String, | ||
age: u8, | ||
} | ||
|
||
// Function hello_rust_cargo without manglng | ||
#[no_mangle] | ||
pub extern "C" fn hello_rust_cargo_main() { | ||
// Print hello world to stdout | ||
|
||
let john = Person { | ||
name: "John".to_string(), | ||
age: 30, | ||
}; | ||
|
||
let json_str = serde_json::to_string(&john).unwrap(); | ||
println!("{}", json_str); | ||
|
||
let jane = Person { | ||
name: "Jane".to_string(), | ||
age: 25, | ||
}; | ||
|
||
let json_str_jane = serde_json::to_string(&jane).unwrap(); | ||
println!("{}", json_str_jane); | ||
|
||
let json_data = r#" | ||
{ | ||
"name": "Alice", | ||
"age": 28 | ||
}"#; | ||
|
||
let alice: Person = serde_json::from_str(json_data).unwrap(); | ||
println!("Deserialized: {} is {} years old", alice.name, alice.age); | ||
|
||
let pretty_json_str = serde_json::to_string_pretty(&alice).unwrap(); | ||
println!("Pretty JSON:\n{}", pretty_json_str); | ||
|
||
tokio::runtime::Builder::new_current_thread() | ||
.enable_all() | ||
.build() | ||
.unwrap() | ||
.block_on(async { | ||
println!("Hello world from tokio!"); | ||
}); | ||
|
||
loop { | ||
// Do nothing | ||
} | ||
} |
Oops, something went wrong.