mirror of
https://github.com/make-42/hayai.git
synced 2025-01-18 18:47:10 +01:00
140 lines
3.5 KiB
Go
140 lines
3.5 KiB
Go
//go:build linux || freebsd || netbsd || openbsd || illumos
|
|
// +build linux freebsd netbsd openbsd illumos
|
|
|
|
package beeep
|
|
|
|
import (
|
|
"errors"
|
|
"os"
|
|
"syscall"
|
|
"time"
|
|
"unsafe"
|
|
)
|
|
|
|
// Constants
|
|
const (
|
|
// This number represents the fixed frequency of the original PC XT's timer chip, which is approximately 1.193 MHz. This number
|
|
// is divided with the desired frequency to obtain a counter value, that is subsequently fed into the timer chip, tied to the PC speaker.
|
|
clockTickRate = 1193180
|
|
|
|
// linux/kd.h, start sound generation (0 for off)
|
|
kiocsound = 0x4B2F
|
|
|
|
// linux/input-event-codes.h
|
|
evSnd = 0x12 // Event type
|
|
sndTone = 0x02 // Sound
|
|
)
|
|
|
|
var (
|
|
// DefaultFreq - frequency, in Hz, middle A
|
|
DefaultFreq = 440.0
|
|
// DefaultDuration - duration in milliseconds
|
|
DefaultDuration = 200
|
|
)
|
|
|
|
// inputEvent represents linux/input.h event structure.
|
|
type inputEvent struct {
|
|
Time syscall.Timeval // time in seconds since epoch at which event occurred
|
|
Type uint16 // event type
|
|
Code uint16 // event code related to the event type
|
|
Value int32 // event value related to the event type
|
|
}
|
|
|
|
// ioctl system call manipulates the underlying device parameters of special files.
|
|
func ioctl(fd, name, data uintptr) error {
|
|
_, _, e := syscall.Syscall(syscall.SYS_IOCTL, fd, name, data)
|
|
if e != 0 {
|
|
return e
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Beep beeps the PC speaker (https://en.wikipedia.org/wiki/PC_speaker).
|
|
//
|
|
// On Linux it needs permission to access `/dev/tty0` or `/dev/input/by-path/platform-pcspkr-event-spkr` files for writing,
|
|
// and `pcspkr` module must be loaded. User must be in correct groups, usually `input` and/or `tty`.
|
|
//
|
|
// If it can not open device files, it will fallback to sending Bell character (https://en.wikipedia.org/wiki/Bell_character).
|
|
// For bell character in X11 terminals you can enable bell with `xset b on`. For console check `setterm` and `--blength` or `--bfreq` options.
|
|
//
|
|
// On macOS this just sends bell character. Enable `Audible bell` in Terminal --> Preferences --> Settings --> Advanced.
|
|
//
|
|
// On Windows it uses Beep function via syscall.
|
|
//
|
|
// On Web it plays hard coded beep sound.
|
|
func Beep(freq float64, duration int) error {
|
|
if freq == 0 {
|
|
freq = DefaultFreq
|
|
} else if freq > 20000 {
|
|
freq = 20000
|
|
} else if freq < 0 {
|
|
freq = DefaultFreq
|
|
}
|
|
|
|
if duration == 0 {
|
|
duration = DefaultDuration
|
|
}
|
|
|
|
period := int(float64(clockTickRate) / freq)
|
|
|
|
var evdev bool
|
|
|
|
f, err := os.OpenFile("/dev/tty0", os.O_WRONLY, 0644)
|
|
if err != nil {
|
|
e := err
|
|
f, err = os.OpenFile("/dev/input/by-path/platform-pcspkr-event-spkr", os.O_WRONLY, 0644)
|
|
if err != nil {
|
|
e = errors.New("beeep: " + e.Error() + "; " + err.Error())
|
|
|
|
// Output the only beep we can
|
|
_, err = os.Stdout.Write([]byte{7})
|
|
if err != nil {
|
|
return errors.New(e.Error() + "; " + err.Error())
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
evdev = true
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
if evdev { // Use Linux evdev API
|
|
ev := inputEvent{}
|
|
ev.Type = evSnd
|
|
ev.Code = sndTone
|
|
ev.Value = int32(freq)
|
|
|
|
d := *(*[unsafe.Sizeof(ev)]byte)(unsafe.Pointer(&ev))
|
|
|
|
// Start beep
|
|
f.Write(d[:])
|
|
|
|
time.Sleep(time.Duration(duration) * time.Millisecond)
|
|
|
|
ev.Value = 0
|
|
d = *(*[unsafe.Sizeof(ev)]byte)(unsafe.Pointer(&ev))
|
|
|
|
// Stop beep
|
|
f.Write(d[:])
|
|
} else { // Use ioctl
|
|
// Start beep
|
|
err = ioctl(f.Fd(), kiocsound, uintptr(period))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
time.Sleep(time.Duration(duration) * time.Millisecond)
|
|
|
|
// Stop beep
|
|
err = ioctl(f.Fd(), kiocsound, uintptr(0))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|