| // Copyright 2023 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package main |
| |
| import ( |
| "bytes" |
| "go/format" |
| "os" |
| "path/filepath" |
| "strings" |
| "testing" |
| ) |
| |
| func TestDLLFilenameEscaping(t *testing.T) { |
| tests := []struct { |
| name string |
| filename string |
| }{ |
| {"no escaping necessary", "kernel32"}, |
| {"escape period", "windows.networking"}, |
| {"escape dash", "api-ms-win-wsl-api-l1-1-0"}, |
| } |
| for _, tt := range tests { |
| t.Run(tt.name, func(t *testing.T) { |
| // Write a made-up syscall into a temp file for testing. |
| const prefix = "package windows\n//sys Example() = " |
| const suffix = ".Example" |
| name := filepath.Join(t.TempDir(), "syscall.go") |
| if err := os.WriteFile(name, []byte(prefix+tt.filename+suffix), 0666); err != nil { |
| t.Fatal(err) |
| } |
| |
| // Ensure parsing, generating, and formatting run without errors. |
| // This is good enough to show that escaping is working. |
| src, err := ParseFiles([]string{name}) |
| if err != nil { |
| t.Fatal(err) |
| } |
| var buf bytes.Buffer |
| if err := src.Generate(&buf); err != nil { |
| t.Fatal(err) |
| } |
| if _, err := format.Source(buf.Bytes()); err != nil { |
| t.Log(buf.String()) |
| t.Fatal(err) |
| } |
| }) |
| } |
| } |
| |
| func TestSyscallNGeneration(t *testing.T) { |
| tests := []struct { |
| name string |
| wantsysfunc string |
| sig string |
| }{ |
| { |
| name: "syscall with 2 params", |
| wantsysfunc: "syscall.SyscallN", |
| sig: "Example(a1 *uint16, a2 *uint16) = ", |
| }, |
| { |
| name: "syscall with 6 params", |
| wantsysfunc: "syscall.SyscallN", |
| sig: "Example(a1 *uint, a2 *uint, a3 *uint, a4 *uint, a5 *uint, a6 *uint) = ", |
| }, |
| { |
| name: "syscall with 15 params", |
| wantsysfunc: "syscall.SyscallN", |
| sig: strings.ReplaceAll(`Example(a1 *uint, a2 *uint, a3 *uint, a4 *uint, a5 *uint, a6 *uint, |
| a7 *uint, a8 *uint, a9 *uint, a10 *uint, a11 *uint, a12 *uint, |
| a13 *uint, a14 *uint, a15 *uint) = `, "\n", ""), |
| }, |
| { |
| name: "syscall with 18 params", |
| wantsysfunc: "syscall.SyscallN", |
| sig: strings.ReplaceAll(`Example(a1 *uint, a2 *uint, a3 *uint, a4 *uint, a5 *uint, a6 *uint, |
| a7 *uint, a8 *uint, a9 *uint, a10 *uint, a11 *uint, a12 *uint, |
| a13 *uint, a14 *uint, a15 *uint, a16 *uint, a17 *uint, a18 *uint) = `, "\n", ""), |
| }, |
| } |
| for _, tt := range tests { |
| t.Run(tt.name, func(t *testing.T) { |
| // Write the syscall into a temp file for testing. |
| prefix := "package windows\n//sys " + tt.sig |
| suffix := ".Example" |
| name := filepath.Join(t.TempDir(), "syscall.go") |
| if err := os.WriteFile(name, []byte(prefix+"example"+suffix), 0666); err != nil { |
| t.Fatal(err) |
| } |
| |
| // Ensure parsing, generating, and formatting run without errors. |
| // This is good enough to show that escaping is working. |
| src, err := ParseFiles([]string{name}) |
| if err != nil { |
| t.Fatal(err) |
| } |
| var buf bytes.Buffer |
| if err := src.Generate(&buf); err != nil { |
| t.Fatal(err) |
| } |
| if _, err := format.Source(buf.Bytes()); err != nil { |
| t.Fatal(err) |
| } |
| |
| if !strings.Contains(buf.String(), tt.wantsysfunc+"(") { |
| t.Fatalf("expected syscall func %q in buffer %s", tt.wantsysfunc, buf.String()) |
| } |
| }) |
| } |
| } |