| | |
| | |
| | |
| |
|
| | |
| |
|
| | |
| | |
| | |
| | |
| | |
| |
|
| | package main |
| |
|
| | import ( |
| | "bytes" |
| | "flag" |
| | "fmt" |
| | "go/format" |
| | "log" |
| | "os" |
| | "slices" |
| | "unicode" |
| | ) |
| |
|
| | var filename = flag.String("output", "isprint.go", "output file name") |
| |
|
| | var ( |
| | range16 []uint16 |
| | except16 []uint16 |
| | range32 []uint32 |
| | except32 []uint32 |
| | ) |
| |
|
| | func isPrint(r rune) bool { |
| | |
| | |
| | |
| | |
| | |
| |
|
| | if 0 <= r && r < 1<<16 { |
| | rr, rang, except := uint16(r), range16, except16 |
| | i, _ := slices.BinarySearch(rang, rr) |
| | if i >= len(rang) || rr < rang[i&^1] || rang[i|1] < rr { |
| | return false |
| | } |
| | _, found := slices.BinarySearch(except, rr) |
| | return !found |
| | } |
| |
|
| | rr, rang, except := uint32(r), range32, except32 |
| | i, _ := slices.BinarySearch(rang, rr) |
| | if i >= len(rang) || rr < rang[i&^1] || rang[i|1] < rr { |
| | return false |
| | } |
| | _, found := slices.BinarySearch(except, rr) |
| | return !found |
| | } |
| |
|
| | func scan(min, max rune) (rang, except []uint32) { |
| | lo := rune(-1) |
| | for i := min; ; i++ { |
| | if (i > max || !unicode.IsPrint(i)) && lo >= 0 { |
| | |
| | if i+1 <= max && unicode.IsPrint(i+1) { |
| | except = append(except, uint32(i)) |
| | continue |
| | } |
| | rang = append(rang, uint32(lo), uint32(i-1)) |
| | lo = -1 |
| | } |
| | if i > max { |
| | break |
| | } |
| | if lo < 0 && unicode.IsPrint(i) { |
| | lo = i |
| | } |
| | } |
| | return |
| | } |
| |
|
| | func to16(x []uint32) []uint16 { |
| | var y []uint16 |
| | for _, v := range x { |
| | if uint32(uint16(v)) != v { |
| | panic("bad 32->16 conversion") |
| | } |
| | y = append(y, uint16(v)) |
| | } |
| | return y |
| | } |
| |
|
| | func main() { |
| | flag.Parse() |
| |
|
| | rang, except := scan(0, 0xFFFF) |
| | range16 = to16(rang) |
| | except16 = to16(except) |
| | range32, except32 = scan(0x10000, unicode.MaxRune) |
| |
|
| | for i := rune(0); i <= unicode.MaxRune; i++ { |
| | if isPrint(i) != unicode.IsPrint(i) { |
| | log.Fatalf("%U: isPrint=%v, want %v\n", i, isPrint(i), unicode.IsPrint(i)) |
| | } |
| | } |
| |
|
| | var buf bytes.Buffer |
| |
|
| | fmt.Fprintf(&buf, `// Copyright 2013 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.`+"\n\n") |
| | fmt.Fprintf(&buf, "// Code generated by go run makeisprint.go -output isprint.go; DO NOT EDIT.\n\n") |
| | fmt.Fprintf(&buf, "package strconv\n\n") |
| |
|
| | fmt.Fprintf(&buf, "// (%d+%d+%d)*2 + (%d)*4 = %d bytes\n\n", |
| | len(range16), len(except16), len(except32), |
| | len(range32), |
| | (len(range16)+len(except16)+len(except32))*2+ |
| | (len(range32))*4) |
| |
|
| | fmt.Fprintf(&buf, "var isPrint16 = []uint16{\n") |
| | for i := 0; i < len(range16); i += 2 { |
| | fmt.Fprintf(&buf, "\t%#04x, %#04x,\n", range16[i], range16[i+1]) |
| | } |
| | fmt.Fprintf(&buf, "}\n\n") |
| |
|
| | fmt.Fprintf(&buf, "var isNotPrint16 = []uint16{\n") |
| | for _, r := range except16 { |
| | fmt.Fprintf(&buf, "\t%#04x,\n", r) |
| | } |
| | fmt.Fprintf(&buf, "}\n\n") |
| |
|
| | fmt.Fprintf(&buf, "var isPrint32 = []uint32{\n") |
| | for i := 0; i < len(range32); i += 2 { |
| | fmt.Fprintf(&buf, "\t%#06x, %#06x,\n", range32[i], range32[i+1]) |
| | } |
| | fmt.Fprintf(&buf, "}\n\n") |
| |
|
| | fmt.Fprintf(&buf, "var isNotPrint32 = []uint16{ // add 0x10000 to each entry\n") |
| | for _, r := range except32 { |
| | if r >= 0x20000 { |
| | log.Fatalf("%U too big for isNotPrint32\n", r) |
| | } |
| | fmt.Fprintf(&buf, "\t%#04x,\n", r-0x10000) |
| | } |
| | fmt.Fprintf(&buf, "}\n\n") |
| |
|
| | |
| | fmt.Fprintf(&buf, "// isGraphic lists the graphic runes not matched by IsPrint.\n") |
| | fmt.Fprintf(&buf, "var isGraphic = []uint16{\n") |
| | for r := rune(0); r <= unicode.MaxRune; r++ { |
| | if unicode.IsPrint(r) != unicode.IsGraphic(r) { |
| | |
| | if !unicode.IsGraphic(r) { |
| | log.Fatalf("%U is printable but not graphic\n", r) |
| | } |
| | if r > 0xFFFF { |
| | log.Fatalf("%U too big for isGraphic\n", r) |
| | } |
| | fmt.Fprintf(&buf, "\t%#04x,\n", r) |
| | } |
| | } |
| | fmt.Fprintf(&buf, "}\n") |
| |
|
| | data, err := format.Source(buf.Bytes()) |
| | if err != nil { |
| | log.Fatal(err) |
| | } |
| | err = os.WriteFile(*filename, data, 0644) |
| | if err != nil { |
| | log.Fatal(err) |
| | } |
| | } |
| |
|