# Copyright (c) 2021-2022 Huawei Device Co., Ltd. # Licensed 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. --- definitions: - name: PandaAssembly template: > .language PandaAssembly - name: r_A template: | .record A {} .function void A.constructor(A a0) { return.void } - name: r_B template: | .record B {} .function void B.constructor(B a0) { return.void } tests: - file-name: call.virt.short.negative isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] title: Object calls description: > Call indicated object method, i.e. create new frame, pass values of arguments and continue execution from the first instruction of a method. Callee should treat accumulator value as undefined and cannot use it until accumulator definition in the new frame. Result (if any) is returned in accumulator (see 'Calling sequence' chapter for more details). Method, its class and the number of argument is resolved by given method_id in runtime constant-pool based on object reference using language-specific semantics (currently only PandaAssembly virtual methods are supported, further extensions are TBD). Object reference is passed in the first source register, arguments are passed starting from the second source register in the same order as in method signature. Non-range instructions can be used to pass up to 4 arguments (including object reference). Unused register slot values will be discarded and corresponding registers will not be passed to the callee). For methods with more arguments range kinds of instruction are to be used, which takes the needed number of arguments starting from 'vs' register (including object reference). verification: - method_id_non_static - compatible_arguments - method_id_accessible exceptions: - x_null - x_abstract commands: - file-name: wrong_object_1 bugid: ['2287', '5271'] isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - compatible_arguments header-template: [r_A, r_B] description: Check incorrect usage of `call.virt.short` instruction. First argument is of incorrect type instead of object reference. runner-options: ['verifier-failure', 'verifier-config'] tags: ['verifier'] code-template: | .function i32 A.foo(A a0%s) { ldai 1 return } .function i32 main() { %s call.virt.short A.foo%s # check object is equal to original movi v0, 1 jne v0, exit_failure ldai 0 return exit_failure: ldai 1 return check-type: none cases: - values: - '' - '' - ',' runner-options: [compile-failure] bugid: ['1855'] - values: - '' - 'movi v0, 0' - ', v0' - values: - '' - | # lda.type A sta.obj v0 - ', v0' bugid: ['2256'] - values: - '' - | # lda.type B sta.obj v0 - ', v0' - values: - '' - 'movi.64 v0, 0' - ', v0' - values: - '' - 'fmovi.64 v0, 0' - ', v0' - values: - '' - | # lda.str "some string" sta.obj v0 - ', v0' - values: - '' - | # lda.str "some string" sta.obj v0 movi v1, 1 - ', v0, v1' - values: - '' - | # lda.str "some string" sta.obj v0 movi.64 v1, 1 - ', v0, v1' - values: - '' - | # lda.str "some string" sta.obj v0 fmovi.64 v1, 1 - ', v0, v1' - values: - ', i32 a1' - | # movi v0, 0 movi v1, 1 - ', v0, v1' - values: - ', i64 a1' - | # movi v0, 0 movi.64 v1, 1 - ', v0, v1' - values: - ', f64 a1' - | # movi v0, 0 fmovi.64 v1, 1.1 - ', v0, v1' - values: - ', i32 a1' - | # movi.64 v0, 0 movi v1, 1 - ', v0, v1' - values: - ', i64 a1' - | # movi.64 v0, 0 movi.64 v1, 1 - ', v0, v1' - values: - ', f64 a1' - | # movi.64 v0, 0 fmovi.64 v1, 1.1 - ', v0, v1' - values: - ', i32 a1' - | # fmovi.64 v0, 0.0 movi v1, 1 - ', v0, v1' - values: - ', i64 a1' - | # fmovi.64 v0, 0 movi.64 v1, 1 - ', v0, v1' - values: - ', f64 a1' - | # fmovi.64 v0, 0 fmovi.64 v1, 1.1 - ', v0, v1' - file-name: j_wrong_object_2 isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - compatible_arguments header-template: [PandaAssembly, r_A, r_B] description: > Check incorrect usage of `call.virt.short` instruction. First argument is a reference to object of incompatible type. Use PandaAssembly language context. runner-options: [use-pa, verifier-failure, verifier-config] tags: [verifier, pa-verifier] bugid: ['1324', '3293', '5271', '6886'] template-cases: - values: - | %s .function i32 A.foo(A a0%s) { ldai 1 return } - values: - | %s ##- %s .function i32 A.foo(A a0) { ldai 1 return } code-template: | %s .function i32 main() { initobj.short B.constructor sta.obj v0 *s call.virt.short A.foo, v0*s check-type: exit-positive cases: - values: - '' - '' - '' - '' - values: - '' - ',i32 a1' - 'movi v1, 0' - ', v1' - values: - '' - ',i32 a1' - 'movi v1, 1' - ', v1' - values: - '' - ', i64 a1' - 'movi.64 v1, 0' - ', v1' - values: - '' - ', i64 a1' - 'movi.64 v1, 0x100000000' - ', v1' - values: - '' - ', f64 a1' - 'fmovi.64 v1, 0' - ', v1' - values: - '' - ', f64 a1' - 'fmovi.64 v1, 1.1' - ', v1' - values: - '.record panda.String ' - ', panda.String a1' - | # lda.str "some string" sta.obj v1 - ', v1' - file-name: p_wrong_object_2 isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - compatible_arguments header-template: [r_A, r_B] description: > Check incorrect usage of `call.virt.short` instruction. First argument is a reference to object of incompatible type. Use PandaAssembly language context. runner-options: ['verifier-failure', 'verifier-config'] tags: [verifier] bugid: ['1324'] template-cases: - values: - | %s .function i32 A.foo(A a0%s) { ldai 1 return } - values: - | %s ##- %s .function i32 A.foo(A a0) { ldai 1 return } code-template: | %s .function i32 main() { initobj.short B.constructor sta.obj v0 *s call.virt.short A.foo, v0*s check-type: exit-positive cases: - values: - '' - '' - '' - '' - values: - '' - ',i32 a1' - 'movi v1, 0' - ', v1' - values: - '' - ',i32 a1' - 'movi v1, 1' - ', v1' - values: - '' - ', i64 a1' - 'movi.64 v1, 0' - ', v1' - values: - '' - ', i64 a1' - 'movi.64 v1, 0x100000000' - ', v1' - values: - '' - ', f64 a1' - 'fmovi.64 v1, 0' - ', v1' - values: - '' - ', f64 a1' - 'fmovi.64 v1, 1.1' - ', v1' - values: - '.record panda.String ' - ', panda.String a1' - | # lda.str "some string" sta.obj v1 - ', v1' - file-name: arg_types isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - compatible_arguments header-template: [r_A] description: > Verifier should report when register type does not match function argument type. code-template: | .record panda.Object .record panda.String .function void A.func(A a0, %s) { return.void } .function i32 main() { initobj A.constructor sta.obj v0 *s call.virt.short A.func, v0, v1 template-cases: - values: - 'i32 a1' exclude: [i32] - values: - 'i64 a1' exclude: [i64] - values: - 'f64 a1' exclude: [f64] - values: - 'i32[] a1' exclude: [i32arr] - values: - 'i64[] a1' exclude: [i64arr] - values: - 'f64[] a1' exclude: [f64arr] - values: - 'panda.String a1' exclude: [str] - values: - 'panda.String[] a1' exclude: [strarr] - values: - 'panda.Object a1' exclude: [obj, objarr, i32arr, i64arr, f64arr, str, strarr] - values: - 'panda.Object[] a1' exclude: [objarr, strarr] check-type: exit-positive bugid: ['1324'] tags: [verifier] runner-options: ['verifier-failure', 'verifier-config'] cases: - values: - | # movi v1, 0 id: i32 - values: - | # movi v1, 1 id: i32 - values: - | # movi.64 v1, 0 id: i64 - values: - | # movi.64 v1, 1 id: i64 - values: - | # fmovi.64 v1, 0 id: f64 - values: - | # fmovi.64 v1, 3.1415926535 id: f64 - values: - | # movi v1, 123 newarr v1, v1, i32[] id: i32arr - values: - | # movi v1, 123 newarr v1, v1, i64[] id: i64arr - values: - | # movi v1, 123 newarr v1, v1, f64[] id: f64arr - values: - | # lda.str "some string" sta.obj v1 id: str - values: - | # movi v1, 123 newarr v1, v1, panda.String[] id: strarr - values: - | # newobj v1, panda.Object id: obj - values: - | # movi v1, 123 newarr v1, v1, panda.Object[] id: objarr - file-name: acc_uninitialized isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] header-template: [r_A] description: > Verifier should report that uninitialized accumulator is used in function. check-type: exit-positive bugid: ['1324'] tags: [verifier] runner-options: ['verifier-failure', 'verifier-config'] code-template: | .function void A.func(A a0) { %s return.void } .function i32 main() { initobj A.constructor sta.obj v0 call.virt.short A.func, v0 cases: - values: - sta v0 - values: - sta.64 v0 - values: - sta.obj v0 - file-name: reg_uninitialized isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] header-template: [r_A] description: > Verifier should report that uninitialized register is used in function. check-type: exit-positive bugid: ['1324'] tags: [verifier] runner-options: ['verifier-failure', 'verifier-config'] code-template: | .function void A.func(A a0) { %s return.void } .function i32 main() { initobj A.constructor sta.obj v0 call.virt.short A.func, v0 template-cases: - values: - lda %s exclude: [r16] - values: - lda.64 %s exclude: [r16] - values: - lda.obj %s exclude: [r16] - values: - mov v0, %s cases: - values: [v0] - values: [v1] - values: [v128] - values: [v255] - values: [v256] id: r16 - values: [v32768] id: r16 - values: [v65535] id: r16 - file-name: p_method_id_accessible isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - method_id_accessible header-template: [r_A] description: > Verifier should report that method is not accessible if method_id is not resolved to existing method. Use PandaAssembly language context. check-type: exit-positive tags: [verifier] runner-options: ['verifier-failure', 'verifier-config'] code-template: | .function %s A.func(A a0, *s a1) .function i32 main() { initobj A.constructor sta.obj v0 *s call.virt.short A.func, v0, v1 template-cases: - values: - 'void' - values: - 'i32' exclude: [void] - values: - 'i64' exclude: [void] - values: - 'f64' exclude: [void] - values: - 'i32[]' exclude: [void] - values: - 'i64[]' exclude: [void] - values: - 'f64[]' exclude: [void] - values: - 'A' exclude: [void] - values: - 'A[]' exclude: [void] cases: - case-template: | .function void A.func(A a0) .function i32 main() { initobj A.constructor sta.obj v0 call.virt.short A.func, v0 id: void - values: ['i32', 'movi v1, 12345678'] - values: ['i64', 'movi.64 v1, 0x123456789ABCDEF'] - values: ['f64', 'fmovi.64 v1, 3.1415926535'] - values: - 'i32[]' - | # movi v1, 123 newarr v1, v1, i32[] - values: - 'i64[]' - | # movi v1, 123 newarr v1, v1, i64[] - values: - 'f64[]' - | # movi v1, 123 newarr v1, v1, f64[] - values: - 'A' - | # initobj A.constructor sta.obj v1 - values: - 'A[]' - | # movi v1, 123 newarr v1, v1, A[] - file-name: j_method_id_accessible isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - method_id_accessible header-template: [PandaAssembly, r_A] description: > Verifier should report that method is not accessible if method_id is not resolved to existing method. Use PandaAssembly language context. check-type: exit-positive bugid: ['3293', '6886'] tags: [verifier, release, pa-verifier] runner-options: [use-pa, verifier-failure, verifier-config] code-template: | %s .record B {} .function void B.constructor(B a0) { return.void } .function i32 main() { initobj B.constructor sta.obj v0 movi v1, 123 call.virt.short *s.func, v0, v1 template-cases: - values: - | .function void A.func(A a0, i32 a1) .function void B.func(B a0, i32 a1) - values: - | .function void A.func(A a0, i32 a1) .function void B.func(B a0, i32 a1) ignore: true bugid: ['3247'] - values: - | .function void A.func(A a0, i32 a1) .function void B.func(B a0, i32 a1) exclude: [a] ignore: true bugid: ['3247'] cases: - values: - A id: a - values: - B id: b - file-name: p_method_id_non_static bugid: ["1324"] isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - method_id_non_static header-template: [r_A] description: > Verifier should report that method is not accessible if method_id is not resolved to existing method. Use PandaAssembly language context. check-type: exit-positive tags: [verifier] runner-options: ['verifier-failure', 'verifier-config'] code-template: | .function %s A.func(A a0, *s a1) { %s } .function i32 main() { initobj A.constructor sta.obj v0 *s call.virt.short A.func, v0, v1 template-cases: - values: - 'void' - 'return.void' - values: - 'i32' - | # ldai 1 return exclude: [void] - values: - 'i64' - | # ldai.64 1 return.64 exclude: [void] - values: - 'f64' - | # fldai.64 3.1415926535 return.64 exclude: [void] - values: - 'i32[]' - | # movi v1, 123 newarr v1, v1, i32[] lda.obj v1 return.obj exclude: [void] - values: - 'i64[]' - | # movi v1, 123 newarr v1, v1, i64[] lda.obj v1 return.obj exclude: [void] - values: - 'f64[]' - | # movi v1, 123 newarr v1, v1, f64[] lda.obj v1 return.obj exclude: [void] - values: - 'A' - | # initobj A.constructor sta.obj v1 return.obj - values: - 'A[]' - | # movi v1, 123 newarr v1, v1, A[] lda.obj v1 return.obj cases: - case-template: | .function void A.func(A a0) { return.void } .function i32 main() { initobj A.constructor sta.obj v0 call.virt.short A.func, v0 id: void - values: ['i32', 'movi v1, 12345678'] - values: ['i64', 'movi.64 v1, 0x123456789ABCDEF'] - values: ['f64', 'fmovi.64 v1, 3.1415926535'] - values: - 'i32[]' - | # movi v1, 123 newarr v1, v1, i32[] - values: - 'i64[]' - | # movi v1, 123 newarr v1, v1, i64[] - values: - 'f64[]' - | # movi v1, 123 newarr v1, v1, f64[] - values: - 'A' - | # initobj A.constructor sta.obj v1 - values: - 'A[]' - | # movi v1, 123 newarr v1, v1, A[] - file-name: j_method_id_non_static tags: [verifier, pa-verifier] bugid: ['1324', '3293', '5271', '6886'] isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] verification: - method_id_non_static header-template: [PandaAssembly, r_A] description: > Verifier should report that method is not accessible if method_id is not resolved to existing method. Use PandaAssembly language context. check-type: exit-positive runner-options: [use-pa, verifier-failure, verifier-config] code-template: | *s .record B {} .function void B.constructor(B a0) { return.void } .function i32 main() { initobj B.constructor sta.obj v0 movi v1, 123 call.virt.short %s.func, v0, v1 template-cases: - values: - A exclude: [virta] # call.virt.short A.func where A.func is virtual - values: - B exclude: [virtb] cases: - values: - | .function void A.func(A a0, i32 a1) { return.void } .function void B.func(B a0, i32 a1) { return.void } - values: - | .function void A.func(A a0, i32 a1) { return.void } .function void B.func(B a0, i32 a1) { return.void } id: virta - values: - | .function void A.func(A a0, i32 a1) { return.void } .function void B.func(B a0, i32 a1) { return.void } id: virtb - file-name: match_parameters_amount isa: instructions: - sig: call.virt.short method_id, v1:in:top, v2:in:top acc: out:top format: [op_v1_4_v2_4_id_16] header-template: [r_A] description: > Compiler should check amount of function parameters with corresponding call.virt.short check-type: none runner-options: [compile-failure] ignore: true bugid: ['1956','1304'] code-template: | .function void A.func(A a0, i32 a1%s) .function i32 main() { call.virt.short A.func, v0%s cases: - values: ['', ''] - values: [', i32 a2', ', v1'] - values: [', i32 a2, i32 a3', ', v1'] - values: [', i32 a2, i32 a3, i32 a4', ', v1']