This is an archive of the discontinued LLVM Phabricator instance.

[WebAssembly] `AsmTypeCheck` support to br instr
AbandonedPublic

Authored by HerrCai0907 on Apr 3 2023, 11:19 PM.

Details

Summary

This PR introduces the BrStack member to store the info about
loop, block, if and try. It can check whether br immediate number
out of range and whether type stack is mismatched in br instr.

Diff Detail

Event Timeline

HerrCai0907 created this revision.Apr 3 2023, 11:19 PM
Herald added a project: Restricted Project. · View Herald TranscriptApr 3 2023, 11:19 PM
HerrCai0907 requested review of this revision.Apr 3 2023, 11:19 PM
Herald added a project: Restricted Project. · View Herald TranscriptApr 3 2023, 11:19 PM

remove unrelated file

sbc100 added a comment.Apr 4 2023, 8:46 AM
This comment was removed by sbc100.

I like this idea. @aheejin is probably also a good reviewer for this.
Could delegate be checked similarly here too?

llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
176

You're calling Operand.getImm() several times here, maybe pull it out into a variable?

add basis support for delegate

Could delegate be checked similarly here too?

Thinks for mentioning. I have fixed the behavior of end block to avoid incorrect BrStack.

Principle it is similarly expect the behavior in catch block. I think it can be done in another patch.

HerrCai0907 marked an inline comment as done.Apr 4 2023, 7:00 PM
HerrCai0907 edited the summary of this revision. (Show Details)

This looks pretty good!

llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
206–207

Should this be outside the loop over the params? Otherwise we start with the same stack when checking each parameter, right?

llvm/test/MC/WebAssembly/type-checker-control-flow.s
83

It would be good to add a newline here.

llvm/test/MC/WebAssembly/type-checker-errors.s
524–525

This is actually valid. The rule for br is that if the label has type [t*], the stack before the br can be [t1* t*] for any t1*. The br implicitly drops everything in t1*.

See https://webassembly.github.io/spec/core/valid/instructions.html#xref-syntax-instructions-syntax-instr-control-mathsf-br-l

br will only check the stack top instead of whole stack

HerrCai0907 marked 3 inline comments as done.Apr 5 2023, 10:44 PM

This will have conflicts with D147852, so depending on which lands first, the other needs to resolve them. D147852 doesn't intend to add new functionalities (like br checking here) though; it just handles other block-ending instructions in the same way as end_block.

llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
51–62

I think you can move this to the header, given that this doesn't have a body

60

I don't think we need to link the formal spec here; it's assumed the implementer and the reader of this file either knows or is able to look for it. We don't link it elsewhere in the file either.

215–216
230–232
233–234
235

Shouldn't we do -1? If not, the message can be something like

br: invalid depth 3 (max 3)

But -1 also has a problem, because when BrStack is empty it will print something like

br: invalid depth 0 (max -1)

So if we want to print (max N), it would be better to handle that case smoothly too.

238–239

Why is this and what does this code do?

238–243

I think you applied clang-format to the whole file. While we try to keep files clang-formatted, I think it'd be better not to mix this with other functionality changes. Can you revert this?

Tip: git clang-format main will clang-format only the changed lines, not the whole file. You can replace main with another branch or a commit if necessary.

240
250
252

TODO aheejin copy? do we use this later again?

281

Please revert unrelated clang-format changes

397–401

Please revert unrelated clang-format changes

413–415

Please revert unrelated clang-format changes

462–464

While is this here, while other block-like instructions (block, try, and loop) are checked separately (line 391-402) above?

llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.h
41
llvm/test/MC/WebAssembly/basic-assembly.s
84–88 ↗(On Diff #511525)

Why is this change necessary?

aheejin added inline comments.Apr 9 2023, 6:23 AM
llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
252

Sorry, forgot to remove my own memo here, and Phabricator doesn't give a way to delete inline comments... Please ignore the message above. By the way, a question: this looks copying the stack intentionally not to modify BrStack.back().EnterStack, correct?

aheejin added inline comments.Apr 9 2023, 6:26 AM
llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
428–432

I don't think we should restore try's EnterStack here? In a catch body, the only input params are the caught values, not the try's input types.

HerrCai0907 planned changes to this revision.Apr 9 2023, 8:09 AM
HerrCai0907 marked an inline comment as done.

I will change those point and add comment also in code.

llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
235

I also think it is -1, but when I use MDN's web runner for following code

(module
  (import "console" "log" (func $log (param i32)))

  (func $hhh
    i32.const 1
    call $log
    br 0
    i32.const 2
    call $log
  )

  (start 1) ;; run the first function automatically
)
``
It outputs `1`
I am not sure whether func body is a block which can also `br` to?
238–239

consider this case

i32.const 1
if  // store op stack
    return
    // here can do everything because of stack-polymorphic
else
    // here stack should be recovered as stored
end_if
238–243

Could it possible to submit a format patch before it?
Actually my arc always does format when I use arc diff.

252

correct. here just checks whether the parameters of label are fulfilled.
e.g.

i32.const 1
block (i32, i32) -> (i32)
end_block
llvm/test/MC/WebAssembly/basic-assembly.s
84–88 ↗(On Diff #511525)

The old file is invalid but not be detected. (The op stack of true and false branch in if block are not balanced)
The reason I drop and push i32 is to avoid coupling in each checking block.
drop all operands in "switch" and re-introduce for "if"

HerrCai0907 added inline comments.Apr 9 2023, 11:32 AM
llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
428–432

The semantic of catch instructions in try [T1->T2] catch [T3] is [T2 -> T3].
If we don't restore the stack, we will get a incorrect stack after end_try.
For example

.tagtype TAG f32
i32.const 1
try i32
    f32.const 1.0
    throw TAG
    // here ok because unreachable
    drop
catch TAG
    i32.trunc_f32_s
end_try
// here type stack is [i32, i32] but without restore we will get error
HerrCai0907 marked 8 inline comments as done.Apr 9 2023, 11:58 AM
HerrCai0907 added inline comments.
llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
428–432
block [T1] 
    xxxxx
end_block
try [T1->T2]
    instr1*
catch [T3]
    instr2*
end_try

after instr1*, the type stack should [T2] and without restore we push T3. After instr2, we will get [T2, T2] which is not incorrect

update without reverting format

D147852 landed, so please rebase onto main and resolve conflicts.

llvm/lib/Target/WebAssembly/AsmParser/WebAssemblyAsmTypeCheck.cpp
60

Stray ;

235

Yes, the function body is also a block you can br to.

238–239

(The comments have moved so it's hard to track what code this was about, but it was about this code)

// In some cases (unreachable / br), `Stack` will be invalid and need to be
// recovered.
if (Unreachable)
  Stack = BrStack.pop_back_val().applySignature();
else
  BrStack.pop_back();

I think we already check this case? (The checker/parser had some bugs but some of them were fixed in D147852 and D147881, so please check them too) Can you give me an example that the current checker/parser misses on this case?

Also it's a little confusing that how much of the functionalities this CL adds overlaps with the existing ones. Currently we do some type checking in the parser too, which is hard to remove because the parser needs to check some basic things to parse. The CL header claims it is adding br checking, and I don't think we do br checking elsewhere, so I think it's a good thing to add, but this CL adds a lot more things. Can we possibly simplify this CL by adding only the missing checks?

238–243

Yeah a separate formatting CL is fine, but I wonder why your arc diff unconditionally formats it. There should be a way or an option not to do it. My arc diff asks me, doesn't unconditionally formats files, when submitting a CL, and also it only asks me if the lines I modified are not formatted and doesn't check other lines. What's your arc version?

417–418

D147852 moves this clause into else if (Name == "end_block" || Name == "end_loop" || Name == "end_if" || ..., so please resolve the conflict

462–464

Ping

HerrCai0907 abandoned this revision.Apr 11 2023, 2:38 PM

simplify checker and re-submit in D148054