| // META: title=validation tests for WebNN API gemm operation |
| // META: global=window |
| // META: variant=?cpu |
| // META: variant=?gpu |
| // META: variant=?npu |
| // META: script=../resources/utils_validation.js |
| |
| 'use strict'; |
| |
| const label = 'gemm_xxx'; |
| const kExampleInputDescriptor = { |
| dataType: 'float32', |
| shape: [2, 2] |
| }; |
| |
| validateTwoInputsFromMultipleBuilders('gemm'); |
| validateTwoBroadcastableInputsTensorLimit('gemm', label); |
| |
| multi_builder_test(async (t, builder, otherBuilder) => { |
| const cFromOtherBuilder = otherBuilder.input('c', kExampleInputDescriptor); |
| const options = {c: cFromOtherBuilder}; |
| |
| const a = builder.input('a', kExampleInputDescriptor); |
| const b = builder.input('b', kExampleInputDescriptor); |
| assert_throws_js(TypeError, () => builder.gemm(a, b, options)); |
| }, '[gemm] throw if c option is from another builder'); |
| |
| const tests = [ |
| { |
| name: '[gemm] Test building gemm with default option.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [3, 4]}, |
| output: {dataType: 'float32', shape: [2, 4]} |
| }, |
| { |
| name: |
| '[gemm] Throw if inputShapeA[1] is not equal to inputShapeB[0] default options.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [2, 4]}, |
| options: {label} |
| }, |
| { |
| name: '[gemm] Test building gemm with aTranspose=true.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [2, 4]}, |
| options: { |
| aTranspose: true, |
| }, |
| output: {dataType: 'float32', shape: [3, 4]} |
| }, |
| { |
| name: |
| '[gemm] Throw if inputShapeA[0] is not equal to inputShapeB[0] with aTranspose=true.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [3, 4]}, |
| options: { |
| aTranspose: true, |
| label: label, |
| }, |
| }, |
| { |
| name: '[gemm] Test building gemm with bTranspose=true.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [4, 3]}, |
| options: { |
| bTranspose: true, |
| }, |
| output: {dataType: 'float32', shape: [2, 4]} |
| }, |
| { |
| name: |
| '[gemm] Throw if inputShapeA[0] is not equal to inputShapeB[0] with bTranspose=true.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [3, 4]}, |
| options: { |
| bTranspose: true, |
| label: label, |
| }, |
| }, |
| { |
| name: '[gemm] Throw if the rank of inputA is not 2.', |
| a: {dataType: 'float32', shape: [2, 3, 1]}, |
| b: {dataType: 'float32', shape: [2, 4]}, |
| options: {label} |
| }, |
| { |
| name: '[gemm] Throw if the rank of inputB is not 2.', |
| a: {dataType: 'float32', shape: [2, 4]}, |
| b: {dataType: 'float32', shape: [2, 3, 1]}, |
| options: {label} |
| }, |
| { |
| name: '[gemm] Throw if data types of two inputs do not match.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float16', shape: [3, 4]}, |
| options: {label} |
| }, |
| { |
| name: '[gemm] Test building gemm with inputC.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [3, 4]}, |
| options: { |
| c: {dataType: 'float32', shape: [4]}, |
| }, |
| output: {dataType: 'float32', shape: [2, 4]} |
| }, |
| { |
| name: '[gemm] Test building gemm with scalar inputC.', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [3, 4]}, |
| options: { |
| c: {dataType: 'float32', shape: []}, |
| }, |
| output: {dataType: 'float32', shape: [2, 4]} |
| }, |
| { |
| name: |
| '[gemm] Throw if inputShapeC is not unidirectionally broadcastable to the output shape [inputShapeA[0], inputShapeB[1]].', |
| a: {dataType: 'float32', shape: [2, 3]}, |
| b: {dataType: 'float32', shape: [3, 4]}, |
| options: { |
| c: {dataType: 'float32', shape: [2, 3]}, |
| label: label, |
| }, |
| }, |
| { |
| name: '[gemm] Throw if the input data type is not floating point.', |
| a: {dataType: 'int32', shape: [2, 3]}, |
| b: {dataType: 'int32', shape: [3, 4]}, |
| options: {label} |
| }, |
| { |
| name: |
| '[gemm] Throw if data type of inputC does not match ones of inputA and inputB.', |
| a: {dataType: 'float32', shape: [3, 2]}, |
| b: {dataType: 'float32', shape: [4, 3]}, |
| options: { |
| c: {dataType: 'float16', shape: [2, 4]}, |
| aTranspose: true, |
| bTranspose: true, |
| label: label, |
| }, |
| }, |
| { |
| name: '[gemm] Throw if the rank of inputC is 3.', |
| a: {dataType: 'float32', shape: [3, 2]}, |
| b: {dataType: 'float32', shape: [4, 3]}, |
| options: { |
| c: {dataType: 'float32', shape: [2, 3, 4]}, |
| aTranspose: true, |
| bTranspose: true, |
| label: label, |
| }, |
| }, |
| ]; |
| |
| tests.forEach( |
| test => promise_test(async t => { |
| const builder = new MLGraphBuilder(context); |
| const a = builder.input('a', test.a); |
| const b = builder.input('b', test.b); |
| if (test.options && test.options.c) { |
| test.options.c = builder.input('c', test.options.c); |
| } |
| if (test.output) { |
| const output = builder.gemm(a, b, test.options); |
| assert_equals(output.dataType, test.output.dataType); |
| assert_array_equals(output.shape, test.output.shape); |
| } else { |
| const regrexp = new RegExp('\\[' + label + '\\]'); |
| assert_throws_with_label( |
| () => builder.gemm(a, b, test.options), regrexp); |
| } |
| }, test.name)); |