Skip to content

Commit

Permalink
examples: New app to build Rust with Cargo
Browse files Browse the repository at this point in the history
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 <huangqi3@xiaomi.com>
  • Loading branch information
no1wudi committed Dec 23, 2024
1 parent 83ea915 commit 055e551
Show file tree
Hide file tree
Showing 8 changed files with 254 additions and 0 deletions.
62 changes: 62 additions & 0 deletions examples/hello_rust_cargo/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
# ##############################################################################
# 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.
#
# ##############################################################################

include(nuttx_add_library)

if(CONFIG_EXAMPLES_HELLO_RUST_CARGO)

# Call cargo build from CMakeLists and add it to the build system. Notice we
# should call cargo build with add_custom_target, otherwise cargo will be
# called every time when cmake is configured.

add_custom_target(
cargo_build_hello_rust ALL
COMMAND
cargo build --release -Zbuild-std=std,panic_abort --manifest-path
${CMAKE_CURRENT_SOURCE_DIR}/hello/Cargo.toml --target
riscv32imac-unknown-nuttx-elf --target-dir
${CMAKE_BINARY_DIR}/hello_rust_cargo
BYPRODUCTS
${CMAKE_BINARY_DIR}/hello_rust_cargo/riscv32imac-unknown-nuttx-elf/release/libhello.a
)

add_custom_target(
hello_rust_cargo
DEPENDS
${CMAKE_BINARY_DIR}/hello_rust_cargo/riscv32imac-unknown-nuttx-elf/release/libhello.a
)

add_dependencies(apps cargo_build_hello_rust) # Add this line

# Call cargo_build target each time when cmake is configured or generated.

nuttx_add_extra_library(${CMAKE_BINARY_DIR}/hello_rust_cargo/riscv32imac-unknown-nuttx-elf/release/libhello.a)

nuttx_add_application(
NAME
${CONFIG_EXAMPLES_HELLO_RUST_CARGO_PROGNAME}
SRCS
proxy_main.c
STACKSIZE
${CONFIG_EXAMPLES_HELLO_STACKSIZE}
PRIORITY
${CONFIG_EXAMPLES_HELLO_PRIORITY})

endif() # CONFIG_EXAMPLES_HELLO_RUST_CARGO
29 changes: 29 additions & 0 deletions examples/hello_rust_cargo/Kconfig
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
23 changes: 23 additions & 0 deletions examples/hello_rust_cargo/Make.defs
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
############################################################################
# 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.
#
############################################################################

ifneq ($(CONFIG_EXAMPLES_HELLO_RUST_CARGO),)
CONFIGURED_APPS += $(APPDIR)/examples/hello_rust_cargo
endif
32 changes: 32 additions & 0 deletions examples/hello_rust_cargo/Makefile
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
############################################################################
# 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)

# Do not suppport building this application from Makefile.

include $(APPDIR)/Application.mk
1 change: 1 addition & 0 deletions examples/hello_rust_cargo/hello/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
target
20 changes: 20 additions & 0 deletions examples/hello_rust_cargo/hello/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
[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"
45 changes: 45 additions & 0 deletions examples/hello_rust_cargo/hello/src/lib.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
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 rust_main() {
// Print hello world to stdout
println!("Hello, world! from Rust and Cargo!");

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);
}
42 changes: 42 additions & 0 deletions examples/hello_rust_cargo/proxy_main.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
/****************************************************************************
* apps/examples/hello_rust_cargo/proxy_main.c
*
* 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.
*
****************************************************************************/

/****************************************************************************
* Included Files
****************************************************************************/

#include <nuttx/config.h>
#include <stdio.h>

/****************************************************************************
* Public Functions
****************************************************************************/

extern void rust_main(void);

/****************************************************************************
* main
****************************************************************************/

int main(int argc, FAR char *argv[])
{
rust_main();
return 0;
}

0 comments on commit 055e551

Please sign in to comment.