blob: 7b50a4d211227b3ea51421fc347f4762998b3102 [file] [log] [blame]
// 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())
}
})
}
}