Compare commits

..

No commits in common. "96f7636f019b9ce85d26afc61de4a889bb033a56" and "141125d932a9d12859be33b7ee986db26339eba5" have entirely different histories.

6 changed files with 187 additions and 220 deletions

3
.gitignore vendored
View File

@ -1,3 +1,2 @@
*.jpg *.jpg
*.png *.png
output/

188
blend.go
View File

@ -8,9 +8,145 @@ import (
"log" "log"
"math" "math"
"os" "os"
"path" "runtime"
"sync"
) )
// job passed to goroutines. blend color from img1 and img2 at position (x, y)
type blendColorJob struct {
X int
Y int
Img1 image.Image
Img2 image.Image
}
// new color after blend, to apply at position (x, y)
type blendColorResult struct {
X int
Y int
Color color.Color
}
type blendWorkerPool struct {
CpuCores int
Jobs chan blendColorJob
Results chan blendColorResult
WorkerWG *sync.WaitGroup
ConsumerWG *sync.WaitGroup
}
func (p *blendWorkerPool) InitWorkerPool() {
p.CpuCores = runtime.NumCPU()
p.Jobs = make(chan blendColorJob, 1)
p.Results = make(chan blendColorResult, 1)
p.WorkerWG = new(sync.WaitGroup)
p.ConsumerWG = new(sync.WaitGroup)
}
// runs the pool of goroutines
func (p *blendWorkerPool) RunWorkers(outImg *image.RGBA) {
for i := 1; i < p.CpuCores; i++ {
p.WorkerWG.Add(1)
p.ConsumerWG.Add(1)
go p.BlendWorker()
go p.SetImageWorker(outImg)
}
}
// blending pixels and assigning them to the output image
func (p *blendWorkerPool) SendBlendJobs(dimensions image.Rectangle, img1 image.Image, img2 image.Image) {
for x := 0; x < dimensions.Max.X; x++ {
for y := 0; y < dimensions.Max.Y; y++ {
p.Jobs <- blendColorJob{x, y, img1, img2}
}
}
close(p.Jobs)
}
// goroutine to blend colors at position (x, y) from two images
func (p *blendWorkerPool) BlendWorker() {
defer p.WorkerWG.Done()
for j := range p.Jobs {
c1 := j.Img1.At(j.X, j.Y)
c2 := j.Img2.At(j.X, j.Y)
c3 := blendColor(c1, c2)
p.Results <- blendColorResult{j.X, j.Y, c3}
}
}
// goroutine to set the new color values at newImage position (x, y)
func (p *blendWorkerPool) SetImageWorker(newImage *image.RGBA) {
defer p.ConsumerWG.Done()
for r := range p.Results {
newImage.Set(r.X, r.Y, r.Color)
}
}
func (p *blendWorkerPool) BlendImagesMain() {
log.Println("Blending the images...")
p.InitWorkerPool()
// output image, ready to receive pixel values
dimensions := dimensionsToRectangle(ConfigRegister.OutputWidth, ConfigRegister.OutputHeight)
outImg := image.NewRGBA(dimensions)
for i := 1; i < ConfigRegister.Amount; i++ {
// get two random images and load them
var img1 image.Image
if ConfigRegister.BaseImage != "" && i == 1 {
img1 = loadImage(ConfigRegister.BaseImage)
} else {
if i != 1 {
img1 = loadImage("output.jpg")
} else {
img1 = loadImage(getRandomImage())
}
}
img2 := loadImage(getRandomImage())
// pool of workers unionizing, ready to blend a new picture using the power of friendship
var pool blendWorkerPool
pool.InitWorkerPool()
// main blending routine
pool.BlendImages(img1, img2, outImg)
encodeImage(outImg)
}
}
func (p *blendWorkerPool) BlendImages(img1 image.Image, img2 image.Image, out *image.RGBA) {
dimensions := dimensionsToRectangle(ConfigRegister.OutputWidth, ConfigRegister.OutputHeight)
// TODO: use a worker pool for those operations ?
// resize image
img1Resized := resize(img1)
img2Resized := resize(img2)
p.RunWorkers(out)
p.SendBlendJobs(dimensions, img1Resized, img2Resized)
// first waitgroup to wait for the results to be ready before closing the channel
p.WorkerWG.Wait()
close(p.Results)
// second waitgroup to ensure the encoding doesn't start before the goroutines are done
p.ConsumerWG.Wait()
}
// encode the image
func encodeImage(imgData *image.RGBA) {
outputFile := fmt.Sprintf("%s/%s", ConfigRegister.OutputDir, "output.jpg")
out, _ := os.Create(outputFile)
defer out.Close()
log.Println("Encoding the new image...")
jpeg.Encode(out, imgData, nil)
log.Println("Done.")
}
// convert RGBA pixel to grayscale. BT.709 luminosity formula // convert RGBA pixel to grayscale. BT.709 luminosity formula
func grayscale(pixel color.Color) color.Color { func grayscale(pixel color.Color) color.Color {
c := color.RGBAModel.Convert(pixel).(color.RGBA) c := color.RGBAModel.Convert(pixel).(color.RGBA)
@ -114,12 +250,46 @@ func blendColor(color1 color.Color, color2 color.Color) color.Color {
return filter(new) return filter(new)
} }
// encode the image // creates a new rectangle with the min height and width from both images
func encodeImage(imgData *image.RGBA) { func getMaxDimensions(img1 image.Image, img2 image.Image) image.Rectangle {
outputFile := fmt.Sprintf("%s/%s", path.Clean(ConfigRegister.OutputDir), "output.jpg") // get dimensions for both images
out, _ := os.Create(outputFile) size1 := img1.Bounds().Size()
defer out.Close() size2 := img2.Bounds().Size()
log.Println("Encoding the new image...")
jpeg.Encode(out, imgData, nil) // final image sized from lowest width and height
log.Println("Done.") width := int(math.Min(float64(size1.X), float64(size2.X)))
height := int(math.Min(float64(size1.Y), float64(size2.Y)))
// the dimensions, as Points, of the output image
upLeft := image.Point{0, 0}
lowRight := image.Point{width, height}
return image.Rectangle{upLeft, lowRight}
}
// resizes image using dimensions from config register (nearest neighbour interpolation)
func resize(img image.Image) (resized *image.RGBA) {
imgSize := img.Bounds().Size()
dimensions := dimensionsToRectangle(ConfigRegister.OutputWidth, ConfigRegister.OutputHeight)
xscale := float64(imgSize.X) / float64(dimensions.Max.X)
yscale := float64(imgSize.Y) / float64(dimensions.Max.Y)
// creates new rescaled image based on a given image dimensions
resized = image.NewRGBA(dimensions)
// get pixels from the original image
for x := 0; x < dimensions.Max.X; x++ {
for y := 0; y < dimensions.Max.Y; y++ {
xp := int(math.Floor(float64(x) * xscale))
yp := int(math.Floor(float64(y) * yscale))
pixel := img.At(xp, yp)
resized.Set(x, y, pixel)
}
}
return
}
// returns a Rectangle of dimensions <width>x<height>
func dimensionsToRectangle(width int, height int) image.Rectangle {
upLeft := image.Point{0, 0}
lowRight := image.Point{width, height}
return image.Rectangle{upLeft, lowRight}
} }

View File

@ -4,7 +4,6 @@ import (
"encoding/json" "encoding/json"
"flag" "flag"
"fmt" "fmt"
"log"
"os" "os"
"strconv" "strconv"
"strings" "strings"
@ -49,35 +48,13 @@ func (c *Config) SetOutputDimensions() {
c.OutputHeight = height c.OutputHeight = height
} }
func (c *Config) CheckPaths() {
_, err := os.Stat(c.OutputDir)
if err != nil {
log.Fatal("Output dir does not exist :", c.OutputDir)
}
_, err = os.Stat(c.InputDir)
if err != nil {
log.Fatal("Input dir does not exist :", c.InputDir)
}
if (c.BaseImage != "") {
_, err = os.Stat(c.BaseImage)
if err != nil {
log.Fatal("Base image does not exist :", c.BaseImage)
}
}
}
func initConfigRegister() { func initConfigRegister() {
// default seed for the RNG // default seed for the RNG
seed := time.Now().UnixNano() seed := time.Now().UnixNano()
// command line arguments // command line arguments
flag.StringVar(&ConfigRegister.Method, "blending", "darken", "Blending methods : darken, lighten, average, fuckyfun") flag.StringVar(&ConfigRegister.Method, "blending", "darken", "Blending methods : darken, lighten, average, fuckyfun")
flag.StringVar(&ConfigRegister.OutputDir, "output", "./output", "Output directory") flag.StringVar(&ConfigRegister.OutputDir, "output", "./", "Output directory")
flag.StringVar(&ConfigRegister.InputDir, "input", "/home/gator/Photos", "Input directory. Where to look the images from") flag.StringVar(&ConfigRegister.InputDir, "input", "/home/gator/Photos", "Input directory. Where to look the images from")
flag.StringVar(&ConfigRegister.BaseImage, "base-img", "", "Path to the base image to work with. Random image if not set") flag.StringVar(&ConfigRegister.BaseImage, "base-img", "", "Path to the base image to work with. Random image if not set")
flag.StringVar(&ConfigRegister.Dimensions, "dimensions", "1280x1024", "Out image dimensions. <width>x<height>") flag.StringVar(&ConfigRegister.Dimensions, "dimensions", "1280x1024", "Out image dimensions. <width>x<height>")
@ -89,8 +66,6 @@ func initConfigRegister() {
flag.Parse() flag.Parse()
ConfigRegister.CheckPaths()
// set output's width and height // set output's width and height
ConfigRegister.SetOutputDimensions() ConfigRegister.SetOutputDimensions()
} }

View File

@ -1,50 +0,0 @@
package main
import (
"image"
"math"
)
// creates a new rectangle with the min height and width from both images
func getMaxDimensions(img1 image.Image, img2 image.Image) image.Rectangle {
// get dimensions for both images
size1 := img1.Bounds().Size()
size2 := img2.Bounds().Size()
// final image sized from lowest width and height
width := int(math.Min(float64(size1.X), float64(size2.X)))
height := int(math.Min(float64(size1.Y), float64(size2.Y)))
// the dimensions, as Points, of the output image
upLeft := image.Point{0, 0}
lowRight := image.Point{width, height}
return image.Rectangle{upLeft, lowRight}
}
// resizes image using dimensions from config register (nearest neighbour interpolation)
func resize(img image.Image) (resized *image.RGBA) {
imgSize := img.Bounds().Size()
dimensions := dimensionsToRectangle(ConfigRegister.OutputWidth, ConfigRegister.OutputHeight)
xscale := float64(imgSize.X) / float64(dimensions.Max.X)
yscale := float64(imgSize.Y) / float64(dimensions.Max.Y)
// creates new rescaled image based on a given image dimensions
resized = image.NewRGBA(dimensions)
// get pixels from the original image
for x := 0; x < dimensions.Max.X; x++ {
for y := 0; y < dimensions.Max.Y; y++ {
xp := int(math.Floor(float64(x) * xscale))
yp := int(math.Floor(float64(y) * yscale))
pixel := img.At(xp, yp)
resized.Set(x, y, pixel)
}
}
return
}
// returns a Rectangle of dimensions <width>x<height>
func dimensionsToRectangle(width int, height int) image.Rectangle {
upLeft := image.Point{0, 0}
lowRight := image.Point{width, height}
return image.Rectangle{upLeft, lowRight}
}

View File

@ -1,6 +1,8 @@
package main package main
import ( import (
_ "image/jpeg"
_ "image/png"
"io/ioutil" "io/ioutil"
"log" "log"
"math/rand" "math/rand"
@ -19,6 +21,9 @@ func main() {
rand.Seed(ConfigRegister.Seed) rand.Seed(ConfigRegister.Seed)
log.Println("Seed :", ConfigRegister.Seed) log.Println("Seed :", ConfigRegister.Seed)
// pool of workers unionizing, ready to blend a new picture using the power of friendship
var pool blendWorkerPool
// launch main blending routine // launch main blending routine
BlendImagesMain() pool.BlendImagesMain()
} }

View File

@ -1,132 +0,0 @@
package main
import (
"image"
"image/color"
"log"
"runtime"
"sync"
)
// job passed to goroutines. blend color from img1 and img2 at position (x, y)
type blendColorJob struct {
X int
Y int
Img1 image.Image
Img2 image.Image
}
// new color after blend, to apply at position (x, y)
type blendColorResult struct {
X int
Y int
Color color.Color
}
type blendWorkerPool struct {
CpuCores int
Jobs chan blendColorJob
Results chan blendColorResult
WorkerWG *sync.WaitGroup
ConsumerWG *sync.WaitGroup
}
func (p *blendWorkerPool) InitWorkerPool() {
p.CpuCores = runtime.NumCPU()
p.Jobs = make(chan blendColorJob, 1)
p.Results = make(chan blendColorResult, 1)
p.WorkerWG = new(sync.WaitGroup)
p.ConsumerWG = new(sync.WaitGroup)
}
// runs the pool of goroutines
func (p *blendWorkerPool) RunWorkers(outImg *image.RGBA) {
for i := 1; i < p.CpuCores; i++ {
p.WorkerWG.Add(1)
p.ConsumerWG.Add(1)
go p.BlendWorker()
go p.SetImageWorker(outImg)
}
}
// blending pixels and assigning them to the output image
func (p *blendWorkerPool) SendBlendJobs(dimensions image.Rectangle, img1 image.Image, img2 image.Image) {
for x := 0; x < dimensions.Max.X; x++ {
for y := 0; y < dimensions.Max.Y; y++ {
p.Jobs <- blendColorJob{x, y, img1, img2}
}
}
close(p.Jobs)
}
// goroutine to blend colors at position (x, y) from two images
func (p *blendWorkerPool) BlendWorker() {
defer p.WorkerWG.Done()
for j := range p.Jobs {
c1 := j.Img1.At(j.X, j.Y)
c2 := j.Img2.At(j.X, j.Y)
c3 := blendColor(c1, c2)
p.Results <- blendColorResult{j.X, j.Y, c3}
}
}
// goroutine to set the new color values at newImage position (x, y)
func (p *blendWorkerPool) SetImageWorker(newImage *image.RGBA) {
defer p.ConsumerWG.Done()
for r := range p.Results {
newImage.Set(r.X, r.Y, r.Color)
}
}
func (p *blendWorkerPool) BlendImages(img1 image.Image, img2 image.Image, out *image.RGBA) {
dimensions := dimensionsToRectangle(ConfigRegister.OutputWidth, ConfigRegister.OutputHeight)
// TODO: use a worker pool for those operations ?
// resize image
img1Resized := resize(img1)
img2Resized := resize(img2)
p.RunWorkers(out)
p.SendBlendJobs(dimensions, img1Resized, img2Resized)
// first waitgroup to wait for the results to be ready before closing the channel
p.WorkerWG.Wait()
close(p.Results)
// second waitgroup to ensure the encoding doesn't start before the goroutines are done
p.ConsumerWG.Wait()
}
func BlendImagesMain() {
log.Println("Blending the images...")
// output image, ready to receive pixel values
dimensions := dimensionsToRectangle(ConfigRegister.OutputWidth, ConfigRegister.OutputHeight)
outImg := image.NewRGBA(dimensions)
for i := 1; i < ConfigRegister.Amount; i++ {
// get two random images and load them
var img1 image.Image
if ConfigRegister.BaseImage != "" && i == 1 {
img1 = loadImage(ConfigRegister.BaseImage)
} else {
if i != 1 {
img1 = loadImage("output.jpg")
} else {
img1 = loadImage(getRandomImage())
}
}
img2 := loadImage(getRandomImage())
// pool of workers unionizing, ready to blend a new picture using the power of friendship
var pool blendWorkerPool
pool.InitWorkerPool()
// main blending routine
pool.BlendImages(img1, img2, outImg)
encodeImage(outImg)
}
}