blob: 6d564c9f02a3e1650032af74e9a52fd4eeb063e8 [file] [log] [blame]
// Copyright 2023 The Wuffs Authors.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// https://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or https://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
//
// SPDX-License-Identifier: Apache-2.0 OR MIT
//go:build ignore
// +build ignore
package main
// truncate-progressive-jpeg.go saves truncated editions of a JPEG, cutting
// just after every SOS (Start Of Scan) marker, appending an EOI (End Of Image)
// marker after the cut.
//
// Usage: go run truncate-progressive-jpeg.go foo.jpeg
import (
"fmt"
"os"
"strings"
)
func main() {
if err := main1(); err != nil {
os.Stderr.WriteString(err.Error() + "\n")
os.Exit(1)
}
}
func main1() error {
if len(os.Args) < 2 {
return fmt.Errorf("usage: progname filename.jpeg")
}
src, err := os.ReadFile(os.Args[1])
if err != nil {
return err
}
prefix, suffix := os.Args[1], ".jpeg"
if strings.HasSuffix(prefix, ".jpg") {
prefix = prefix[:len(prefix)-4]
suffix = ".jpg"
} else if strings.HasSuffix(prefix, ".jpeg") {
prefix = prefix[:len(prefix)-5]
}
pos := 0
if len(src) < 2 {
return posError(pos)
}
scanNumber := 0
for pos <= (len(src) - 2) {
if src[pos] != 0xFF {
return posError(pos)
}
marker := src[pos+1]
pos += 2
switch {
case (0xD0 <= marker) && (marker < 0xD9):
// RSTn (Restart) and SOI (Start Of Image) markers have no payload.
continue
case marker == 0xD9:
// EOI (End Of Image) marker.
return nil
}
if pos >= (len(src) - 2) {
return posError(pos)
}
payloadLength := (int(src[pos]) << 8) | int(src[pos+1])
pos += payloadLength
if (payloadLength < 2) || (pos < 0) || (pos > len(src)) {
return posError(pos)
}
if marker != 0xDA { // SOS (Start Of Scan) marker.
continue
}
for pos < len(src) {
if src[pos] != 0xFF {
pos += 1
} else if pos >= (len(src) - 1) {
return posError(pos)
} else if m := src[pos+1]; (m != 0x00) && ((m < 0xD0) || (0xD7 < m)) {
break
} else {
pos += 2
}
}
outName := fmt.Sprintf("%s.scan%03d%s", prefix, scanNumber, suffix)
scanNumber++
data := ([]byte)(nil)
data = append(data, src[:pos]...)
data = append(data, "\xFF\xD9"...) // Append an EOI marker.
if err := os.WriteFile(outName, data, 0644); err != nil {
return err
}
fmt.Printf("Wrote %s (cut at 0x%08X = %6d bytes).\n", outName, pos, pos)
}
return posError(pos)
}
func posError(pos int) error {
return fmt.Errorf("invalid JPEG at pos = 0x%08X = %10d", pos, pos)
}