Skip to content

Commit dd8e00f

Browse files
committed
Create a generic AVR target: avr-unknown-unknown
This commit removes the `avr-unknown-gnu-atmega328` target and replaces it with a more generic `avr-unknown-unknown` variant that can be specialized using `-C target-cpu` (e.g. `-C target-cpu=atmega328p`). I've decided to go with the `-unknown` tag (i.e. `avr-unknown-unknown` instead of `avr-unknown-gnu`), because that's the name LLVM already uses - I see no reason to diverge here. Seizing the day, I'm adding myself as the maintainer of this target - I've been already fixing the bugs anyway, might as well make it official. Related discussion: #131171
1 parent ecf2d1f commit dd8e00f

File tree

15 files changed

+130
-59
lines changed

15 files changed

+130
-59
lines changed

compiler/rustc_codegen_ssa/src/back/link.rs

+12-2
Original file line numberDiff line numberDiff line change
@@ -1920,10 +1920,20 @@ fn add_post_link_objects(
19201920

19211921
/// Add arbitrary "pre-link" args defined by the target spec or from command line.
19221922
/// FIXME: Determine where exactly these args need to be inserted.
1923-
fn add_pre_link_args(cmd: &mut dyn Linker, sess: &Session, flavor: LinkerFlavor) {
1923+
fn add_pre_link_args(
1924+
cmd: &mut dyn Linker,
1925+
sess: &Session,
1926+
flavor: LinkerFlavor,
1927+
codegen_results: &CodegenResults,
1928+
) {
19241929
if let Some(args) = sess.target.pre_link_args.get(&flavor) {
19251930
cmd.verbatim_args(args.iter().map(Deref::deref));
19261931
}
1932+
1933+
if sess.target.arch == "avr" {
1934+
cmd.verbatim_arg(format!("-mmcu={}", codegen_results.crate_info.target_cpu));
1935+
}
1936+
19271937
cmd.verbatim_args(&sess.opts.unstable_opts.pre_link_args);
19281938
}
19291939

@@ -2215,7 +2225,7 @@ fn linker_with_args(
22152225
// FIXME: In practice built-in target specs use this for arbitrary order-independent options,
22162226
// introduce a target spec option for order-independent linker options and migrate built-in
22172227
// specs to it.
2218-
add_pre_link_args(cmd, sess, flavor);
2228+
add_pre_link_args(cmd, sess, flavor, codegen_results);
22192229

22202230
// ------------ Object code and libraries, order-dependent ------------
22212231

compiler/rustc_target/src/spec/base/avr_gnu.rs compiler/rustc_target/src/spec/base/avr.rs

-39
Original file line numberDiff line numberDiff line change
@@ -1,44 +1,5 @@
11
use object::elf;
22

3-
use crate::spec::{Cc, LinkerFlavor, Lld, RelocModel, Target, TargetOptions};
4-
5-
/// A base target for AVR devices using the GNU toolchain.
6-
///
7-
/// Requires GNU avr-gcc and avr-binutils on the host system.
8-
/// FIXME: Remove the second parameter when const string concatenation is possible.
9-
pub(crate) fn target(target_cpu: &'static str, mmcu: &'static str) -> Target {
10-
Target {
11-
arch: "avr".into(),
12-
metadata: crate::spec::TargetMetadata {
13-
description: None,
14-
tier: None,
15-
host_tools: None,
16-
std: None,
17-
},
18-
data_layout: "e-P1-p:16:8-i8:8-i16:8-i32:8-i64:8-f32:8-f64:8-n8-a:8".into(),
19-
llvm_target: "avr-unknown-unknown".into(),
20-
pointer_width: 16,
21-
options: TargetOptions {
22-
env: "gnu".into(),
23-
24-
c_int_width: "16".into(),
25-
cpu: target_cpu.into(),
26-
exe_suffix: ".elf".into(),
27-
28-
linker: Some("avr-gcc".into()),
29-
eh_frame_header: false,
30-
pre_link_args: TargetOptions::link_args(LinkerFlavor::Gnu(Cc::Yes, Lld::No), &[mmcu]),
31-
late_link_args: TargetOptions::link_args(LinkerFlavor::Gnu(Cc::Yes, Lld::No), &[
32-
"-lgcc",
33-
]),
34-
max_atomic_width: Some(16),
35-
atomic_cas: false,
36-
relocation_model: RelocModel::Static,
37-
..TargetOptions::default()
38-
},
39-
}
40-
}
41-
423
/// Resolve the value of the EF_AVR_ARCH field for AVR ELF files, given the
434
/// name of the target CPU / MCU.
445
///

compiler/rustc_target/src/spec/base/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
pub(crate) mod aix;
22
pub(crate) mod android;
33
pub(crate) mod apple;
4-
pub(crate) mod avr_gnu;
4+
pub(crate) mod avr;
55
pub(crate) mod bpf;
66
pub(crate) mod dragonfly;
77
pub(crate) mod freebsd;

compiler/rustc_target/src/spec/mod.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ pub use base::apple::{
6363
deployment_target_for_target as current_apple_deployment_target,
6464
platform as current_apple_platform,
6565
};
66-
pub use base::avr_gnu::ef_avr_arch;
66+
pub use base::avr::ef_avr_arch;
6767

6868
/// Linker is called through a C/C++ compiler.
6969
#[derive(Clone, Copy, Debug, Eq, Ord, PartialEq, PartialOrd)]
@@ -1734,7 +1734,7 @@ supported_targets! {
17341734
("riscv64gc-unknown-fuchsia", riscv64gc_unknown_fuchsia),
17351735
("x86_64-unknown-fuchsia", x86_64_unknown_fuchsia),
17361736

1737-
("avr-unknown-gnu-atmega328", avr_unknown_gnu_atmega328),
1737+
("avr-unknown-unknown", avr_unknown_unknown),
17381738

17391739
("x86_64-unknown-l4re-uclibc", x86_64_unknown_l4re_uclibc),
17401740

compiler/rustc_target/src/spec/targets/avr_unknown_gnu_atmega328.rs

-5
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
use crate::spec::{Cc, LinkerFlavor, Lld, RelocModel, Target, TargetOptions};
2+
3+
pub(crate) fn target() -> Target {
4+
Target {
5+
arch: "avr".into(),
6+
metadata: crate::spec::TargetMetadata {
7+
description: None,
8+
tier: None,
9+
host_tools: None,
10+
std: None,
11+
},
12+
data_layout: "e-P1-p:16:8-i8:8-i16:8-i32:8-i64:8-f32:8-f64:8-n8-a:8".into(),
13+
llvm_target: "avr-unknown-unknown".into(),
14+
pointer_width: 16,
15+
options: TargetOptions {
16+
c_int_width: "16".into(),
17+
exe_suffix: ".elf".into(),
18+
linker: Some("avr-gcc".into()),
19+
eh_frame_header: false,
20+
pre_link_args: TargetOptions::link_args(LinkerFlavor::Gnu(Cc::Yes, Lld::No), &[]),
21+
late_link_args: TargetOptions::link_args(LinkerFlavor::Gnu(Cc::Yes, Lld::No), &[
22+
"-lgcc",
23+
]),
24+
max_atomic_width: Some(16),
25+
atomic_cas: false,
26+
relocation_model: RelocModel::Static,
27+
..TargetOptions::default()
28+
},
29+
}
30+
}

src/doc/rustc/src/platform-support.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -293,7 +293,7 @@ target | std | host | notes
293293
[`armv7k-apple-watchos`](platform-support/apple-watchos.md) | ✓ | | Armv7-A Apple WatchOS
294294
[`armv7s-apple-ios`](platform-support/apple-ios.md) | ✓ | | Armv7-A Apple-A6 Apple iOS
295295
[`armv8r-none-eabihf`](platform-support/armv8r-none-eabihf.md) | * | | Bare Armv8-R, hardfloat
296-
`avr-unknown-gnu-atmega328` | * | | AVR. Requires `-Z build-std=core`
296+
`avr-unknown-unknown` | * | | AVR
297297
`bpfeb-unknown-none` | * | | BPF (big endian)
298298
`bpfel-unknown-none` | * | | BPF (little endian)
299299
`csky-unknown-linux-gnuabiv2` | ✓ | | C-SKY abiv2 Linux (little endian)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
# `avr-unknown-unknown`
2+
3+
Series of microcontrollers from Atmel: ATmega8, ATmega328p etc.
4+
5+
**Tier: 3**
6+
7+
## Target maintainers
8+
9+
-[Patryk Wychowaniec](https://github.com/Patryk27) <pwychowaniec@pm.me>
10+
11+
## Requirements
12+
13+
This target is only cross-compiled - x86_64 / aarch64 x Lixux / MacOS hosts are
14+
confirmed to work, but in principle any machine able to run rustc and avr-gcc
15+
should be good.
16+
17+
Compiling for this target requires `avr-gcc`, because a couple of intrinsics
18+
(like 32-bit multiplication) rely on [`libgcc`](https://github.com/gcc-mirror/gcc/blob/3269a722b7a03613e9c4e2862bc5088c4a17cc11/libgcc/config/avr/lib1funcs.S)
19+
and can't be provided through `compiler-builtins` yet; this is a limitation that
20+
we hope to lift in the future.
21+
22+
## Building the target
23+
24+
Rust comes with AVR support enabled, you don't have to rebuild the compiler
25+
itself.
26+
27+
## Building Rust programs
28+
29+
Install `avr-gcc`:
30+
31+
```console
32+
# Ubuntu:
33+
$ sudo apt-get install gcc-avr
34+
35+
# Mac:
36+
$ brew tap osx-cross/avr && brew install avr-gcc
37+
38+
# NixOS (takes a couple of minutes, since Hydra doesn't build it):
39+
$ nix shell nixpkgs#pkgsCross.avr.buildPackages.gcc11
40+
```
41+
42+
... setup `.cargo/config` for your project:
43+
44+
```toml
45+
[build]
46+
target = "avr-unknown-unknown"
47+
rustflags = ["-C", "target-cpu=atmega328p"]
48+
49+
[unstable]
50+
build-std = ["core"]
51+
```
52+
53+
... and then simply run:
54+
55+
```console
56+
$ cargo build --release
57+
```
58+
59+
The final binary will be placed into
60+
`./target/avr-unknown-unknown/release/your-project.elf`.
61+
62+
Note that since AVRs have rather small amounts of registers, ROM and RAM, it's
63+
recommended to always use `--release` to avoid running out of space.
64+
65+
## Testing
66+
67+
You can use [`simavr`](https://github.com/buserror/simavr) to emulate the
68+
resulting firmware on your machine:
69+
70+
```
71+
simavr -m atmega328p ./target/avr-unknown-unknown/release/your-project.elf
72+
```
73+
74+
Alternatively, if you want to write a couple of actual `#[test]`s, you can use
75+
[`avr-tester`](https://github.com/Patryk27/avr-tester).

src/tools/compiletest/src/header/tests.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -380,7 +380,7 @@ fn profiler_runtime() {
380380
#[test]
381381
fn asm_support() {
382382
let asms = [
383-
("avr-unknown-gnu-atmega328", false),
383+
("avr-unknown-unknown", false),
384384
("i686-unknown-netbsd", true),
385385
("riscv32gc-unknown-linux-gnu", true),
386386
("riscv64imac-unknown-none-elf", true),

tests/assembly/asm/avr-modifiers.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
//@ assembly-output: emit-asm
2-
//@ compile-flags: --target avr-unknown-gnu-atmega328
2+
//@ compile-flags: --target avr-unknown-unknown -C target-cpu=atmega328p
33
//@ needs-llvm-components: avr
44

55
#![feature(no_core, lang_items, rustc_attrs, asm_experimental_arch)]

tests/assembly/asm/avr-types.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
//@ assembly-output: emit-asm
2-
//@ compile-flags: --target avr-unknown-gnu-atmega328
2+
//@ compile-flags: --target avr-unknown-unknown -C target-cpu=atmega328p
33
//@ needs-llvm-components: avr
44

55
#![feature(no_core, lang_items, rustc_attrs, asm_experimental_arch)]

tests/assembly/targets/targets-pe.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -15,9 +15,9 @@
1515
//@ revisions: arm64ec_pc_windows_msvc
1616
//@ [arm64ec_pc_windows_msvc] compile-flags: --target arm64ec-pc-windows-msvc
1717
//@ [arm64ec_pc_windows_msvc] needs-llvm-components: aarch64
18-
//@ revisions: avr_unknown_gnu_atmega328
19-
//@ [avr_unknown_gnu_atmega328] compile-flags: --target avr-unknown-gnu-atmega328
20-
//@ [avr_unknown_gnu_atmega328] needs-llvm-components: avr
18+
//@ revisions: avr_unknown_unknown
19+
//@ [avr_unknown_unknown] compile-flags: --target avr-unknown-unknown -C target-cpu=atmega328p
20+
//@ [avr_unknown_unknown] needs-llvm-components: avr
2121
//@ revisions: bpfeb_unknown_none
2222
//@ [bpfeb_unknown_none] compile-flags: --target bpfeb-unknown-none
2323
//@ [bpfeb_unknown_none] needs-llvm-components: bpf

tests/codegen/avr/avr-func-addrspace.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
//@ compile-flags: -O --target=avr-unknown-gnu-atmega328 --crate-type=rlib -C panic=abort
1+
//@ compile-flags: -O --target=avr-unknown-unknown -C target-cpu=atmega328p --crate-type=rlib -C panic=abort
22
//@ needs-llvm-components: avr
33

44
// This test validates that function pointers can be stored in global variables

tests/ui/feature-gates/feature-gate-abi-avr-interrupt.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
//@ needs-llvm-components: avr
2-
//@ compile-flags: --target=avr-unknown-gnu-atmega328 --crate-type=rlib
2+
//@ compile-flags: --target=avr-unknown-unknown -C target-cpu=atmega328p --crate-type=rlib
33
#![no_core]
44
#![feature(no_core, lang_items)]
55
#[lang="sized"]

tests/ui/repr/16-bit-repr-c-enum.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
//@ revisions: avr msp430
33
//
44
//@ [avr] needs-llvm-components: avr
5-
//@ [avr] compile-flags: --target=avr-unknown-gnu-atmega328 --crate-type=rlib
5+
//@ [avr] compile-flags: --target=avr-unknown-unknown -C target-cpu=atmega328p --crate-type=rlib
66
//@ [msp430] needs-llvm-components: msp430
77
//@ [msp430] compile-flags: --target=msp430-none-elf --crate-type=rlib
88
#![feature(no_core, lang_items, intrinsics, staged_api, rustc_attrs)]

0 commit comments

Comments
 (0)