bleh/main.go

615 lines
18 KiB
Go
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
This file is part of Bleh!.
Bleh! is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
Bleh! is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with Foobar. If not, see <https://www.gnu.org/licenses/>.
*/
package main
import (
"flag"
"fmt"
"image"
"image/color"
_ "image/gif"
_ "image/jpeg"
_ "image/png"
"io"
"log"
"os"
"time"
"github.com/disintegration/imaging"
"github.com/go-ble/ble"
dither "github.com/makeworld-the-better-one/dither"
"tinygo.org/x/bluetooth"
)
const minLines = 86 // firmware refuses to print anything shorter
func MustParseUUID(s string) bluetooth.UUID {
uuid, err := bluetooth.ParseUUID(s)
if err != nil {
panic("invalid UUID: " + err.Error())
}
return uuid
}
var (
mainServiceUUID = MustParseUUID("ae30")
printCharacteristic = MustParseUUID("ae01")
notifyCharacteristic = MustParseUUID("ae02")
dataCharacteristic = MustParseUUID("ae03")
targetPrinterName = "MXW01"
scanTimeout = 10 * time.Second
printCommandHeader = []byte{0x22, 0x21}
printCommandFooter = byte(0xFF)
intensity int
mode string
ditherType string
getStatus bool
getBattery bool
getVersion bool
getPrintType bool
getQueryCount bool
ejectPaper uint
retractPaper uint
outputPath string
version = "dev"
)
func init() {
flag.IntVar(&intensity, "intensity", 80, "Print intensity (0-100)")
flag.IntVar(&intensity, "i", 80, "Print intensity (0-100)")
flag.StringVar(&mode, "mode", "1bpp", "Print mode: 1bpp or 4bpp")
flag.StringVar(&mode, "m", "1bpp", "Print mode: 1bpp or 4bpp")
flag.StringVar(&ditherType, "dither", "none", "Dither method: none, floyd, bayer2x2, bayer4x4, bayer8x8, bayer16x16, atkinson, jjn")
flag.StringVar(&ditherType, "d", "none", "Dither method: none, floyd, bayer2x2, bayer4x4, bayer8x8, bayer16x16, atkinson, jjn")
flag.BoolVar(&getStatus, "status", false, "Query printer status")
flag.BoolVar(&getStatus, "s", false, "Query printer status")
flag.BoolVar(&getBattery, "battery", false, "Query battery level")
flag.BoolVar(&getBattery, "b", false, "Query battery level")
flag.BoolVar(&getVersion, "version", false, "Query printer version")
flag.BoolVar(&getVersion, "v", false, "Query printer version")
flag.BoolVar(&getPrintType, "printtype", false, "Query print type")
flag.BoolVar(&getPrintType, "p", false, "Query print type")
flag.BoolVar(&getQueryCount, "querycount", false, "Query internal counter")
flag.BoolVar(&getQueryCount, "q", false, "Query internal counter")
flag.UintVar(&ejectPaper, "eject", 0, "Eject paper by N lines")
flag.UintVar(&ejectPaper, "E", 0, "Eject paper by N lines")
flag.UintVar(&retractPaper, "retract", 0, "Retract paper by N lines")
flag.UintVar(&retractPaper, "R", 0, "Retract paper by N lines")
flag.StringVar(&outputPath, "o", "", "Output PNG preview instead of printing (specify output path)")
flag.StringVar(&outputPath, "output", "", "Output PNG preview instead of printing (specify output path)")
flag.Usage = func() {
fmt.Fprintf(os.Stderr, "Bleh! Cat Printer Utility for MXW01, version %s\n", version)
fmt.Fprintf(os.Stderr, "Usage: %s [options] <image_path or ->\n", os.Args[0])
fmt.Fprintln(os.Stderr, `
Options:
-i, --intensity int Print intensity (0-100) (default 80)
-m, --mode string Print mode: 1bpp or 4bpp (default "1bpp")
-d, --dither string Dither method: none, floyd, bayer2x2, bayer4x4, bayer8x8, bayer16x16, atkinson, jjn (default "none")
-s, --status Query printer status
-b, --battery Query battery level
-v, --version Query printer version
-p, --printtype Query print type
-q, --querycount Query internal counter
-E, --eject uint Eject paper by N lines
-R, --retract uint Retract paper by N lines
-o, --output <file> Output PNG preview instead of printing.
If <file> is "-", writes PNG to stdout.
<image_path or -> Path to PNG/JPG to print, or '-' for stdin`)
}
}
func parseNotification(data []byte) {
if len(data) < 2 || data[0] != 0x22 || data[1] != 0x21 {
fmt.Printf("Invalid notification header, raw: % X", data)
return
}
cmd := data[2]
dataLen := int(data[4]) | int(data[5])<<8
switch cmd {
case 0xA1: // GetStatus
battery := data[9]
temp := data[10]
statusOk := data[12] == 0
statusCode := data[6]
errCode := data[13]
statusMsg := "Unknown"
if statusOk {
switch statusCode {
case 0x0:
statusMsg = "Standby"
case 0x1:
statusMsg = "Printing"
case 0x2:
statusMsg = "Feeding paper"
case 0x3:
statusMsg = "Ejecting paper"
}
} else {
switch errCode {
case 0x1, 0x9:
statusMsg = "No paper"
case 0x4:
statusMsg = "Overheated"
case 0x8:
statusMsg = "Low battery"
}
}
fmt.Printf("Status: %v (%s), Battery: %d, Temp: %d\n", statusOk, statusMsg, battery, temp)
case 0xA3: // EjectPaper
fmt.Println("Ejecting paper...")
case 0xA4: // RetractPaper
fmt.Println("Retracting paper...")
case 0xA7: // QueryCount
if len(data) >= 12 {
fmt.Printf("Query count: % X\n", data[6:12])
}
case 0xA9: // Print
printOk := data[6] == 0
fmt.Printf("Print status: %s\n", map[bool]string{true: "Ok", false: "Failure"}[printOk])
case 0xAA: // PrintComplete
fmt.Println("Printing finished.")
case 0xAB: // BatteryLevel
fmt.Printf("Battery level: %d\n", data[6])
case 0xB0: // GetPrintType
var t string
switch data[6] {
case 0x01:
t = `High pressure`
case 0xFF:
t = `"Unknown`
default:
t = `Low pressure`
}
fmt.Printf("Print type: %s\n", t)
case 0xB1: // GetVersion
if len(data) < 14+dataLen {
fmt.Println("Malformed version notification")
return
}
version := string(data[6 : 6+dataLen])
var t string
switch data[14] {
case 0x32:
t = `High pressure`
case 0x31:
t = `Low pressure`
default:
t = `Unknown`
}
fmt.Printf("Version: %s, Print type: %s\n", version, t)
default:
fmt.Printf("Received notification for unknown command: 0x%02X\n", cmd)
}
}
const (
linePixels = 384
bytesPerLine = linePixels / 8
)
// decodeImage loads an image from a given path or stdin ("-")
func decodeImage(path string) (image.Image, error) {
if path == "-" {
return decodeImageFromReader(os.Stdin)
}
img, err := imaging.Open(path, imaging.AutoOrientation(true))
if err != nil {
return nil, fmt.Errorf("failed to open image %q: %v", path, err)
}
return img, nil
}
// decodeImageFromReader reads and decodes an image from any io.Reader
func decodeImageFromReader(r io.Reader) (image.Image, error) {
img, _, err := image.Decode(r)
if err != nil {
return nil, fmt.Errorf("decode error: %v", err)
}
return img, nil
}
// loadImageMonoFromImage processes an image.Image to 1bpp packed byte format
func loadImageMonoFromImage(img image.Image, ditherType string) ([]byte, int, error) {
ratio := float64(img.Bounds().Dx()) / float64(img.Bounds().Dy())
height := int(float64(linePixels) / ratio)
img = imaging.Resize(img, linePixels, height, imaging.Lanczos)
img = imaging.Grayscale(img)
if ditherType != "none" {
palette := []color.Color{color.Black, color.White}
d := dither.NewDitherer(palette)
switch ditherType {
case "floyd":
d.Matrix = dither.FloydSteinberg
case "bayer2x2":
d.Mapper = dither.Bayer(2, 2, 1.0)
case "bayer4x4":
d.Mapper = dither.Bayer(4, 4, 1.0)
case "bayer8x8":
d.Mapper = dither.Bayer(8, 8, 1.0)
case "bayer16x16":
d.Mapper = dither.Bayer(16, 16, 1.0)
case "atkinson":
d.Matrix = dither.Atkinson
case "jjn":
d.Matrix = dither.JarvisJudiceNinke
default:
return nil, 0, fmt.Errorf("unknown dither type: %s", ditherType)
}
img = d.DitherCopy(img)
} else {
img = imaging.AdjustContrast(img, 10)
}
pixels := make([]byte, (linePixels*height)/8)
for y := 0; y < height; y++ {
for x := 0; x < linePixels; x++ {
gray := color.GrayModel.Convert(img.At(x, y)).(color.Gray)
if gray.Y < 128 {
idx := (y*linePixels + x) / 8
pixels[idx] |= 1 << (x % 8)
}
}
}
return pixels, height, nil
}
// loadImage4BitFromImage processes an image.Image to 4bpp packed byte format
func loadImage4BitFromImage(img image.Image, ditherType string) ([]byte, int, error) {
ratio := float64(img.Bounds().Dx()) / float64(img.Bounds().Dy())
height := int(float64(linePixels) / ratio)
img = imaging.Resize(img, linePixels, height, imaging.Lanczos)
img = imaging.Grayscale(img)
palette := make([]color.Color, 16)
for i := 0; i < 16; i++ {
v := uint8(i * 17)
palette[i] = color.Gray{Y: 255 - v}
}
if ditherType != "none" {
d := dither.NewDitherer(palette)
switch ditherType {
case "floyd":
d.Matrix = dither.FloydSteinberg
case "bayer2x2":
d.Mapper = dither.Bayer(2, 2, 0.2)
case "bayer4x4":
d.Mapper = dither.Bayer(4, 4, 0.2)
case "bayer8x8":
d.Mapper = dither.Bayer(8, 8, 0.2)
case "bayer16x16":
d.Mapper = dither.Bayer(16, 16, 0.2)
case "atkinson":
d.Matrix = dither.Atkinson
case "jjn":
d.Matrix = dither.JarvisJudiceNinke
default:
return nil, 0, fmt.Errorf("unknown dither type: %s", ditherType)
}
img = d.DitherCopy(img)
}
width := linePixels
pixels := make([]byte, (width*height)/2)
for y := 0; y < height; y++ {
for x := 0; x < width; x++ {
gray := color.GrayModel.Convert(img.At(x, y)).(color.Gray)
level := (255 - gray.Y) >> 4 // 0..15, inverted logic
idx := (y*width + x) >> 1
shift := uint(((x & 1) ^ 1) << 2)
pixels[idx] |= level << shift
}
}
return pixels, height, nil
}
// Extend sendImageToPrinter to handle 4-bit mode
type PrintMode byte
const (
Mode1bpp PrintMode = 0x00
Mode4bpp PrintMode = 0x02
)
func sendImageBufferToPrinter(client ble.Client, dataChr, printChr *ble.Characteristic, pixels []byte, height int, mode PrintMode, intensity byte) error {
fmt.Printf("Sending image: %dx%d lines\n", linePixels, height)
cmd := buildCommand(0xA2, []byte{intensity})
if err := client.WriteCharacteristic(printChr, cmd, true); err != nil {
return fmt.Errorf("intensity set failed: %v", err)
}
param := []byte{
byte(height & 0xFF), byte(height >> 8),
0x30,
byte(mode),
}
cmd = buildCommand(0xA9, param)
if err := client.WriteCharacteristic(printChr, cmd, true); err != nil {
return fmt.Errorf("print command failed: %v", err)
}
bytesPerLine := linePixels / 8
if mode == Mode4bpp {
bytesPerLine = linePixels / 2
}
mtu := 20
for y := 0; y < height; y++ {
slice := pixels[y*bytesPerLine : (y+1)*bytesPerLine]
for offset := 0; offset < len(slice); offset += mtu {
end := offset + mtu
if end > len(slice) {
end = len(slice)
}
chunk := slice[offset:end]
if err := client.WriteCharacteristic(dataChr, chunk, true); err != nil {
return fmt.Errorf("line %d chunk write failed: %v", y, err)
}
time.Sleep(6 * time.Millisecond)
}
}
cmd = buildCommand(0xAD, []byte{0x00})
if err := client.WriteCharacteristic(printChr, cmd, true); err != nil {
return fmt.Errorf("flush failed: %v", err)
}
return nil
}
func padImageToMinLines(img image.Image, minLines int) image.Image {
bounds := img.Bounds()
if bounds.Dy() >= minLines {
return img
}
// Create a new white image
dst := imaging.New(bounds.Dx(), minLines, color.White)
// Paste the original image at the top
dst = imaging.Paste(dst, img, image.Pt(0, 0))
return dst
}
func sendSimpleCommand(client ble.Client, printChr *ble.Characteristic, cmdId byte) error {
cmd := buildCommand(cmdId, []byte{0x00})
return client.WriteCharacteristic(printChr, cmd, true)
}
func sendLineCommand(client ble.Client, printChr *ble.Characteristic, cmdId byte, lines uint) error {
param := []byte{byte(lines & 0xFF), byte(lines >> 8)}
cmd := buildCommand(cmdId, param)
return client.WriteCharacteristic(printChr, cmd, true)
}
func renderPreviewFrom1bpp(pixels []byte, width, height int) image.Image {
img := image.NewGray(image.Rect(0, 0, width, height))
for y := 0; y < height; y++ {
for x := 0; x < width; x++ {
idx := (y*width + x) / 8
bit := uint(x % 8)
if pixels[idx]&(1<<bit) != 0 {
img.SetGray(x, y, color.Gray{Y: 0})
} else {
img.SetGray(x, y, color.Gray{Y: 255})
}
}
}
return img
}
func renderPreviewFrom4bpp(pixels []byte, width, height int) image.Image {
img := image.NewGray(image.Rect(0, 0, width, height))
for y := 0; y < height; y++ {
for x := 0; x < width; x++ {
idx := (y*width + x) >> 1
shift := uint(((x & 1) ^ 1) << 2)
val := (pixels[idx] >> shift) & 0x0F
gray := 255 - val*17
img.SetGray(x, y, color.Gray{Y: gray})
}
}
return img
}
var adapter = bluetooth.DefaultAdapter
func initBLE() {
err := adapter.Enable()
if err != nil {
log.Fatalf("failed to enable BLE adapter: %v", err)
}
}
func main() {
flag.Parse()
needNotifications := getStatus || getBattery || getVersion || getPrintType || getQueryCount || ejectPaper > 0 || retractPaper > 0
if needNotifications {
//ctx, stop := signal.NotifyContext(context.Background(), os.Interrupt)
//defer stop()
initBLE()
log.Println("Scanning for printer...")
var printerAddr bluetooth.Address
err := adapter.Scan(func(adapter *bluetooth.Adapter, result bluetooth.ScanResult) {
fmt.Printf("Found device: %s, Name: %s\n", result.Address.String(), result.LocalName())
if result.LocalName() == targetPrinterName {
printerAddr = result.Address
log.Printf("Found target printer: %s", targetPrinterName)
adapter.StopScan()
}
})
if err != nil {
log.Fatalf("Scan error: %v", err)
}
if printerAddr == (bluetooth.Address{}) {
log.Fatal("Printer not found.")
}
log.Println("Connecting...")
device, err := adapter.Connect(printerAddr, bluetooth.ConnectionParams{})
if err != nil {
log.Fatalf("Connect failed: %v", err)
}
log.Println("Connected to printer:", device.Address)
var printChr, notifyChr *bluetooth.DeviceCharacteristic
services, err := device.DiscoverServices([]bluetooth.UUID{mainServiceUUID})
if err != nil || len(services) == 0 {
log.Fatalf("Service discovery failed: %v", err)
}
svc := services[0]
chars, err := svc.DiscoverCharacteristics(nil)
if err != nil {
log.Fatalf("Characteristic discovery failed: %v", err)
}
for i, c := range chars {
switch c.UUID().String() {
case notifyCharacteristic.String():
notifyChr = &chars[i]
case printCharacteristic.String():
printChr = &chars[i]
}
}
if notifyChr != nil {
err = notifyChr.EnableNotifications(func(b []byte) {
parseNotification(b)
})
if err != nil {
log.Printf("Subscribe failed: %v notifications will be ignored", err)
} else {
log.Println("Subscribed to printer notifications.")
}
}
if printChr == nil {
log.Fatal("Print/command characteristic not found!")
}
if getStatus {
printChr.WriteWithoutResponse([]byte{0xA1})
}
if getBattery {
printChr.WriteWithoutResponse([]byte{0xAB})
}
if getVersion {
printChr.WriteWithoutResponse([]byte{0xB1})
}
if getPrintType {
printChr.WriteWithoutResponse([]byte{0xB0})
}
if getQueryCount {
printChr.WriteWithoutResponse([]byte{0xA7})
}
if ejectPaper > 0 {
printChr.WriteWithoutResponse([]byte{0xA3, byte(ejectPaper)})
}
if retractPaper > 0 {
printChr.WriteWithoutResponse([]byte{0xA4, byte(retractPaper)})
}
if getStatus || getBattery || getVersion || getPrintType || getQueryCount || ejectPaper > 0 || retractPaper > 0 {
log.Println("Waiting for notifications...")
time.Sleep(5 * time.Second)
}
if flag.NArg() < 1 {
return // no image to print, other commands may have run
} else if flag.NArg() >= 1 && needNotifications {
log.Fatalf("Refusing to print and query at the same time due to a firmware bug. Please run print and query commands separately.")
}
}
log.Println("Done!")
}
func buildCommand(cmdId byte, payload []byte) []byte {
cmd := append([]byte{}, printCommandHeader...)
cmd = append(cmd, cmdId)
cmd = append(cmd, 0x00) // reserved
cmd = append(cmd, byte(len(payload)&0xFF), byte(len(payload)>>8))
cmd = append(cmd, payload...)
cmd = append(cmd, calculateCRC8(payload))
cmd = append(cmd, printCommandFooter)
return cmd
}
func calculateCRC8(data []byte) byte {
table := [256]byte{
0x00, 0x07, 0x0e, 0x09, 0x1c, 0x1b, 0x12, 0x15,
0x38, 0x3f, 0x36, 0x31, 0x24, 0x23, 0x2a, 0x2d,
0x70, 0x77, 0x7e, 0x79, 0x6c, 0x6b, 0x62, 0x65,
0x48, 0x4f, 0x46, 0x41, 0x54, 0x53, 0x5a, 0x5d,
0xe0, 0xe7, 0xee, 0xe9, 0xfc, 0xfb, 0xf2, 0xf5,
0xd8, 0xdf, 0xd6, 0xd1, 0xc4, 0xc3, 0xca, 0xcd,
0x90, 0x97, 0x9e, 0x99, 0x8c, 0x8b, 0x82, 0x85,
0xa8, 0xaf, 0xa6, 0xa1, 0xb4, 0xb3, 0xba, 0xbd,
0xc7, 0xc0, 0xc9, 0xce, 0xdb, 0xdc, 0xd5, 0xd2,
0xff, 0xf8, 0xf1, 0xf6, 0xe3, 0xe4, 0xed, 0xea,
0xb7, 0xb0, 0xb9, 0xbe, 0xab, 0xac, 0xa5, 0xa2,
0x8f, 0x88, 0x81, 0x86, 0x93, 0x94, 0x9d, 0x9a,
0x27, 0x20, 0x29, 0x2e, 0x3b, 0x3c, 0x35, 0x32,
0x1f, 0x18, 0x11, 0x16, 0x03, 0x04, 0x0d, 0x0a,
0x57, 0x50, 0x59, 0x5e, 0x4b, 0x4c, 0x45, 0x42,
0x6f, 0x68, 0x61, 0x66, 0x73, 0x74, 0x7d, 0x7a,
0x89, 0x8e, 0x87, 0x80, 0x95, 0x92, 0x9b, 0x9c,
0xb1, 0xb6, 0xbf, 0xb8, 0xad, 0xaa, 0xa3, 0xa4,
0xf9, 0xfe, 0xf7, 0xf0, 0xe5, 0xe2, 0xeb, 0xec,
0xc1, 0xc6, 0xcf, 0xc8, 0xdd, 0xda, 0xd3, 0xd4,
0x69, 0x6e, 0x67, 0x60, 0x75, 0x72, 0x7b, 0x7c,
0x51, 0x56, 0x5f, 0x58, 0x4d, 0x4a, 0x43, 0x44,
0x19, 0x1e, 0x17, 0x10, 0x05, 0x02, 0x0b, 0x0c,
0x21, 0x26, 0x2f, 0x28, 0x3d, 0x3a, 0x33, 0x34,
0x4e, 0x49, 0x40, 0x47, 0x52, 0x55, 0x5c, 0x5b,
0x76, 0x71, 0x78, 0x7f, 0x6a, 0x6d, 0x64, 0x63,
0x3e, 0x39, 0x30, 0x37, 0x22, 0x25, 0x2c, 0x2b,
0x06, 0x01, 0x08, 0x0f, 0x1a, 0x1d, 0x14, 0x13,
0xae, 0xa9, 0xa0, 0xa7, 0xb2, 0xb5, 0xbc, 0xbb,
0x96, 0x91, 0x98, 0x9f, 0x8a, 0x8d, 0x84, 0x83,
0xde, 0xd9, 0xd0, 0xd7, 0xc2, 0xc5, 0xcc, 0xcb,
0xe6, 0xe1, 0xe8, 0xef, 0xfa, 0xfd, 0xf4, 0xf3}
crc := byte(0)
for _, b := range data {
crc = table[crc^b]
}
return crc
}