Skip to content

Commit 74ad9b9

Browse files
committed
Add option_env support
gcc/rust/ChangeLog: * expand/rust-macro-builtins-utility.cc: Add macro expansion for option_env with eager expansion * expand/rust-macro-builtins.cc: Add option_env to builtin list * expand/rust-macro-builtins.h: Add option_env handler to header file gcc/testsuite/ChangeLog: * rust/compile/macros/builtin/option_env1.rs: Add success case for option_env * rust/compile/macros/builtin/option_env2.rs: Add failure case for option_env * rust/compile/macros/builtin/option_env3.rs: Add second failure case for option_env * rust/execute/torture/builtin_macro_option_env.rs: Add execution case for option_env Signed-off-by: Liam Naddell <[email protected]>
1 parent be1e78b commit 74ad9b9

File tree

7 files changed

+224
-2
lines changed

7 files changed

+224
-2
lines changed

gcc/rust/expand/rust-macro-builtins-utility.cc

+78-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
// <http://www.gnu.org/licenses/>.
1818

1919
#include "rust-fmt.h"
20+
#include "rust-ast-builder.h"
2021
#include "rust-macro-builtins.h"
2122
#include "rust-macro-builtins-helpers.h"
2223

@@ -226,6 +227,82 @@ MacroBuiltin::env_handler (location_t invoc_locus, AST::MacroInvocData &invoc,
226227
return AST::Fragment ({node}, std::move (tok));
227228
}
228229

230+
/* Expand builtin macro option_env!(), which inspects an environment variable at
231+
compile time. */
232+
tl::optional<AST::Fragment>
233+
MacroBuiltin::option_env_handler (location_t invoc_locus,
234+
AST::MacroInvocData &invoc,
235+
AST::InvocKind semicolon)
236+
{
237+
auto invoc_token_tree = invoc.get_delim_tok_tree ();
238+
MacroInvocLexer lex (invoc_token_tree.to_token_stream ());
239+
Parser<MacroInvocLexer> parser (lex);
240+
241+
auto last_token_id = macro_end_token (invoc_token_tree, parser);
242+
std::unique_ptr<AST::LiteralExpr> lit_expr = nullptr;
243+
bool has_error = false;
244+
245+
auto start = lex.get_offs ();
246+
auto expanded_expr = try_expand_many_expr (parser, last_token_id,
247+
invoc.get_expander (), has_error);
248+
auto end = lex.get_offs ();
249+
250+
auto tokens = lex.get_token_slice (start, end);
251+
252+
if (has_error)
253+
return AST::Fragment::create_error ();
254+
255+
auto pending = check_for_eager_invocations (expanded_expr);
256+
if (!pending.empty ())
257+
return make_eager_builtin_invocation (BuiltinMacro::OptionEnv, invoc_locus,
258+
invoc_token_tree,
259+
std::move (pending));
260+
261+
if (expanded_expr.size () != 1)
262+
{
263+
rust_error_at (invoc_locus, "%<option_env!%> takes 1 argument");
264+
return AST::Fragment::create_error ();
265+
}
266+
267+
if (expanded_expr.size () > 0)
268+
if (!(lit_expr
269+
= try_extract_string_literal_from_fragment (invoc_locus,
270+
expanded_expr[0])))
271+
return AST::Fragment::create_error ();
272+
273+
parser.skip_token (last_token_id);
274+
275+
auto env_value = getenv (lit_expr->as_string ().c_str ());
276+
AST::Builder b (invoc_locus);
277+
278+
if (env_value == nullptr)
279+
{
280+
std::unique_ptr<AST::Expr> none_expr
281+
= std::unique_ptr<AST::Expr> (new AST::PathInExpression (
282+
b.path_in_expression ({"core", "option", "Option", "None"})));
283+
284+
auto node = AST::SingleASTNode (std::move (none_expr));
285+
std::vector<AST::SingleASTNode> nodes;
286+
nodes.push_back (node);
287+
288+
return AST::Fragment (nodes, std::vector<std::unique_ptr<AST::Token>> ());
289+
}
290+
std::vector<std::unique_ptr<AST::Expr>> args;
291+
args.push_back (b.literal_string (env_value));
292+
293+
std::unique_ptr<AST::Expr> some_expr
294+
= b.call (std::unique_ptr<AST::Expr> (new AST::PathInExpression (
295+
b.path_in_expression ({"core", "option", "Option", "Some"}))),
296+
std::move (args));
297+
298+
auto node = AST::SingleASTNode (std::move (some_expr));
299+
300+
std::vector<AST::SingleASTNode> nodes;
301+
nodes.push_back (node);
302+
303+
return AST::Fragment (nodes, std::vector<std::unique_ptr<AST::Token>> ());
304+
}
305+
229306
tl::optional<AST::Fragment>
230307
MacroBuiltin::cfg_handler (location_t invoc_locus, AST::MacroInvocData &invoc,
231308
AST::InvocKind semicolon)
@@ -296,4 +373,4 @@ MacroBuiltin::stringify_handler (location_t invoc_locus,
296373
return AST::Fragment ({node}, std::move (token));
297374
}
298375

299-
} // namespace Rust
376+
} // namespace Rust

gcc/rust/expand/rust-macro-builtins.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -120,8 +120,8 @@ std::unordered_map<std::string, AST::MacroTranscriberFunc>
120120
{"format_args_nl", format_args_maker (AST::FormatArgs::Newline::Yes)},
121121
{"asm", inline_asm_maker (AST::AsmKind::Inline)},
122122
{"global_asm", inline_asm_maker (AST::AsmKind::Global)},
123+
{"option_env", MacroBuiltin::option_env_handler},
123124
/* Unimplemented macro builtins */
124-
{"option_env", MacroBuiltin::sorry},
125125
{"concat_idents", MacroBuiltin::sorry},
126126
{"module_path", MacroBuiltin::sorry},
127127
{"log_syntax", MacroBuiltin::sorry},

gcc/rust/expand/rust-macro-builtins.h

+4
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,10 @@ class MacroBuiltin
159159
AST::MacroInvocData &invoc,
160160
AST::InvocKind semicolon);
161161

162+
static tl::optional<AST::Fragment>
163+
option_env_handler (location_t invoc_locus, AST::MacroInvocData &invoc,
164+
AST::InvocKind semicolon);
165+
162166
static tl::optional<AST::Fragment> cfg_handler (location_t invoc_locus,
163167
AST::MacroInvocData &invoc,
164168
AST::InvocKind semicolon);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
#![feature(rustc_attrs)]
2+
3+
#[rustc_builtin_macro]
4+
macro_rules! option_env {
5+
() => {}
6+
}
7+
8+
#[lang = "sized"]
9+
trait Sized {}
10+
11+
pub mod core {
12+
pub mod option {
13+
pub enum Option<T> {
14+
Some(T),
15+
None,
16+
}
17+
}
18+
}
19+
20+
use core::option::Option;
21+
22+
23+
fn main() {
24+
// Both a guaranteed-to-exist variable and a failed find should compile
25+
let _: Option<&str> = option_env!("PWD");
26+
let _: Option<&str> = option_env!("PROBABLY_DOESNT_EXIST");
27+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
#![feature(rustc_attrs)]
2+
3+
#[rustc_builtin_macro]
4+
macro_rules! option_env {
5+
() => {}
6+
}
7+
8+
#[lang = "sized"]
9+
trait Sized {}
10+
11+
pub mod core {
12+
pub mod option {
13+
pub enum Option<T> {
14+
Some(T),
15+
None,
16+
}
17+
}
18+
}
19+
20+
use core::option::Option;
21+
22+
fn main() {
23+
let _: Option<&str> = option_env!(42);
24+
// { dg-error "argument must be a string literal" "" { target *-*-* } .-1 }
25+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
#![feature(rustc_attrs)]
2+
3+
#[rustc_builtin_macro]
4+
macro_rules! option_env {
5+
() => {}
6+
}
7+
8+
#[lang = "sized"]
9+
trait Sized {}
10+
11+
pub mod core {
12+
pub mod option {
13+
pub enum Option<T> {
14+
Some(T),
15+
None,
16+
}
17+
}
18+
}
19+
20+
use core::option::Option;
21+
22+
23+
fn main() {
24+
let _: Option<&str> = option_env!("A","B");
25+
// { dg-error "'option_env!' takes 1 argument" "" { target *-*-* } .-1 }
26+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
// { dg-output "VALUE\r*\nVALUE\r*\n" }
2+
// { dg-set-compiler-env-var ENV_MACRO_TEST "VALUE" }
3+
4+
#![feature(rustc_attrs)]
5+
6+
#[rustc_builtin_macro]
7+
macro_rules! option_env {
8+
() => {{}};
9+
}
10+
11+
#[lang = "sized"]
12+
trait Sized {}
13+
14+
pub mod core {
15+
pub mod option {
16+
pub enum Option<T> {
17+
Some(T),
18+
None,
19+
}
20+
}
21+
}
22+
23+
use core::option::Option;
24+
25+
extern "C" {
26+
fn printf(fmt: *const i8, ...);
27+
}
28+
29+
fn print(s: &str) {
30+
unsafe {
31+
printf(
32+
"%s\n" as *const str as *const i8,
33+
s as *const str as *const i8,
34+
);
35+
}
36+
}
37+
38+
macro_rules! env_macro_test {
39+
() => { "ENV_MACRO_TEST" }
40+
}
41+
42+
fn main() -> i32 {
43+
let val0: Option<&'static str> = option_env!("ENV_MACRO_TEST");
44+
45+
46+
match val0 {
47+
Option::None => {},
48+
Option::Some(s) => {
49+
print(s);
50+
}
51+
}
52+
53+
//eager expansion test
54+
let val1: Option<&'static str> = option_env!(env_macro_test!(),);
55+
56+
match val1 {
57+
Option::None => {},
58+
Option::Some(s) => {
59+
print(s);
60+
}
61+
}
62+
0
63+
}

0 commit comments

Comments
 (0)