Skip to content
This repository has been archived by the owner on Dec 10, 2023. It is now read-only.

Commit

Permalink
Fix for issue hlorenzi#115: label support inside asm{}
Browse files Browse the repository at this point in the history
  • Loading branch information
Emmett81 committed Oct 22, 2021
1 parent 4453e9b commit 95a6654
Show file tree
Hide file tree
Showing 4 changed files with 225 additions and 95 deletions.
24 changes: 24 additions & 0 deletions converge.asm
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
#ruledef test
{
ld {x} =>
{
assert(x <= 0x8)
0x11 @ x`16
}
ld {x} =>
{
assert(x > 0x8)
0x22 @ x`8
}

test => asm
{
ld label ; error: converge
ld label ; error: converge
ld label ; error: converge
label:
}
}

test
241 changes: 146 additions & 95 deletions src/asm/state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1331,128 +1331,179 @@ impl State
fileserver: &dyn util::FileServer)
-> Result<expr::Value, ()>
{
// Clone the context in order to advance the logical address
// between instructions.
let mut inner_ctx = ctx.clone();

let mut result = util::BigInt::new(0, Some(0));

let mut parser = syntax::Parser::new(Some(info.report.clone()), info.tokens);

//println!("asm block `{}`", fileserver.get_excerpt(&parser.get_full_span()));

while !parser.is_over()
{
// Substitute `{x}` occurrences with tokens from the argument
let mut subs_parser = parser.slice_until_linebreak_over_nested_braces();
let subparser_span = subs_parser.get_full_span();
let mut size_guess = 0;

//println!("> instr `{}`", fileserver.get_excerpt(&subparser_span));
// Parse the tokens twice, once to find the labels and guess
// size of result, then to assemble the output
for iterations in 0..2 {

let mut subs_tokens: Vec<syntax::Token> = Vec::new();
while !subs_parser.is_over()
{
if let Some(open_token) = subs_parser.maybe_expect(syntax::TokenKind::BraceOpen)
{
let arg_name_token = subs_parser.expect(syntax::TokenKind::Identifier)?;
let arg_name = arg_name_token.excerpt.as_ref().unwrap();
// Clone the context in order to advance the logical address
// between instructions.
let mut inner_ctx = ctx.clone();

let token_sub = match info.args.get_token_sub(&arg_name)
{
None =>
{
info.report.error_span("unknown argument", &arg_name_token.span);
return Err(());
}
Some(t) => t
};
let mut parser = syntax::Parser::new(Some(info.report.clone()), info.tokens);

let close_token = subs_parser.expect(syntax::TokenKind::BraceClose)?;
let sub_span = open_token.span.join(&close_token.span);
//println!("asm block `{}`", fileserver.get_excerpt(&parser.get_full_span()));

for token in token_sub
{
let mut sub_token = token.clone();
sub_token.span = sub_span.clone();
subs_tokens.push(sub_token);
}
}
else
{
subs_tokens.push(subs_parser.advance());
}
}

let mut subparser = syntax::Parser::new(Some(info.report.clone()), &subs_tokens);
subparser.suppress_reports();
while !parser.is_over()
{
// Substitute `{x}` occurrences with tokens from the argument
let mut subs_parser = parser.slice_until_linebreak_over_nested_braces();
let subparser_span = subs_parser.get_full_span();

//println!("> after subs `{:?}`", subs_tokens);

let matches = asm::parser::match_rule_invocation(
&self,
subparser,
inner_ctx.clone(),
fileserver,
info.report.clone())?;
//println!("> instr `{}`", fileserver.get_excerpt(&subparser_span));

let value = self.resolve_rule_invocation(
info.report.clone(),
&matches,
fileserver,
true,
info.args)?;

//println!(" value = {:?}", value);

let (bigint, size) = match value.get_bigint()
{
Some(bigint) =>
let mut subs_tokens: Vec<syntax::Token> = Vec::new();
while !subs_parser.is_over()
{
match bigint.size
if let Some(open_token) = subs_parser.maybe_expect(syntax::TokenKind::BraceOpen)
{
Some(size) => (bigint, size),
None =>
let arg_name_token = subs_parser.expect(syntax::TokenKind::Identifier)?;
let arg_name = arg_name_token.excerpt.as_ref().unwrap();

let token_sub = match info.args.get_token_sub(&arg_name)
{
info.report.error_span(
"cannot infer size of instruction",
&subparser_span);
None =>
{
info.report.error_span("unknown argument", &arg_name_token.span);
return Err(());
}
Some(t) => t
};

let close_token = subs_parser.expect(syntax::TokenKind::BraceClose)?;
let sub_span = open_token.span.join(&close_token.span);

return Err(());
for token in token_sub
{
let mut sub_token = token.clone();
sub_token.span = sub_span.clone();
subs_tokens.push(sub_token);
}
}
else
{
subs_tokens.push(subs_parser.advance());
}
}

_ =>
let mut subparser = syntax::Parser::new(Some(info.report.clone()), &subs_tokens);
subparser.suppress_reports();

//println!("> after subs `{:?}`", subs_tokens);

if subparser.next_is(0, syntax::TokenKind::Identifier) && subparser.next_is(1, syntax::TokenKind::Colon)
{
info.report.error_span(
"wrong type returned from instruction",
&subparser_span);
let label_tk = subparser.expect(syntax::TokenKind::Identifier)?;

let label_name = label_tk.excerpt.as_ref().unwrap();

return Err(());
}
};
info.args.set_local(
label_name,
expr::Value::make_integer(
self.get_addr(info.report.clone(), &inner_ctx, &subparser_span)?));

if size > 0
{
if result.size.unwrap() == 0
{
result = bigint;
subparser.expect(syntax::TokenKind::Colon)?;
}
else
else
{
result = result.concat(
(result.size.unwrap(), 0),
&bigint,
(size, 0));
let matches = asm::parser::match_rule_invocation(
&self,
subparser,
inner_ctx.clone(),
fileserver,
info.report.clone())?;

if iterations == 0
{
let value = self.resolve_rule_invocation(
info.report.clone(),
&matches,
fileserver,
false,
info.args).unwrap_or(expr::Value::make_integer(0));

if value.get_bigint().is_some() && value.get_bigint().unwrap().size.is_some() {
let size = value.get_bigint().unwrap().size.unwrap();
size_guess += size;
inner_ctx.bit_offset += size;
}
}

if iterations == 1
{
let value = self.resolve_rule_invocation(
info.report.clone(),
&matches,
fileserver,
true,
info.args)?;

//println!(" value = {:?}", value);

let (bigint, size) = match value.get_bigint()
{
Some(bigint) =>
{
match bigint.size
{
Some(size) => (bigint, size),
None =>
{
info.report.error_span(
"cannot infer size of instruction",
&subparser_span);

return Err(());
}
}
}

_ =>
{
info.report.error_span(
"wrong type returned from instruction",
&subparser_span);

return Err(());
}
};

if size > 0
{
if result.size.unwrap() == 0
{
result = bigint;
}
else
{
result = result.concat(
(result.size.unwrap(), 0),
&bigint,
(size, 0));
}

inner_ctx.bit_offset += size;
}
}
}

inner_ctx.bit_offset += size;
parser.expect_linebreak()?;
}

parser.expect_linebreak()?;
}

//println!(" result size guess = {:?}", size_guess);
//println!(" result size = {:?}", result.size);

if size_guess != result.size.unwrap()
{
info.report.error_span(
"Size of instruction did not converge after iterations",
&syntax::Parser::new(Some(info.report.clone()), info.tokens).get_full_span());
return Err(());
}

Ok(expr::Value::make_integer(result))
}
}
29 changes: 29 additions & 0 deletions tests/issue115/3.asm
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
#ruledef {
emit {x:u8} => x

nested_test => asm
{
label:
emit $
label2:
emit $
emit label
emit label2
emit $
test
}

test => asm
{
label:
emit $
label2:
emit $
emit label
emit label2
emit $
}
}

test ; = 0x00_01_00_01_04
nested_test ; = 0x05_06_05_06_09_0a_0b_0a_0b_0e
26 changes: 26 additions & 0 deletions tests/issue115/4.asm
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
#ruledef {
emit {x:u8} => x

test => asm
{
test2 end
test2 end
end:
}

test2 {l: u32} => asm
{
emit l
emit end
end:
}

run => asm
{
test
emit 0x10
}
}

run ; = 0x04_02_04_04_10
emit $ ; = 0x05

0 comments on commit 95a6654

Please sign in to comment.