1
0
mirror of https://gitlab.com/80486DX2-66/gists synced 2025-01-25 14:51:45 +05:30
gists/js-programming/bytebeat-render.js

138 lines
3.8 KiB
JavaScript
Raw Normal View History

2023-12-29 19:55:56 +03:00
/*
* bytebeat-render.js [Node]
*
* Node.JS: Bytebeat generator
*
* Note: Calls `ffmpeg` to convert output raw audio to a WAVE file
*
* Warning: The current result is quick and dirty. Not for educational or
* production purposes.
*
2024-03-17 19:16:41 +03:00
* TODO: Fix signed bytebeat and floatbeat
2024-07-12 03:12:15 +03:00
* TODO: Add support for arbitrary (1 <= integer <= 2^32 - 1, because of
* WAVE/RIFF format limitations) audio bit depth
*
* Author: Intel A80486DX2-66
* License: Creative Commons Zero 1.0 Universal
*/
2023-12-29 19:55:56 +03:00
const { appendFileSync, unlinkSync, writeFileSync } = require("fs")
const { tmpdir } = require("os")
const { execSync } = require("child_process")
const { basename } = require("path")
let BUFFER_SIZE = 65536 // feel free to change this
2024-01-01 01:42:17 +03:00
const LIGHTNING_MODE = false // disables sequential file write optimization,
// feel free to enable this
2023-12-29 19:55:56 +03:00
const SAMPLE_RATE = 8000 // feel free to change this
const SECONDS = 30 // feel free to change this
const CHANNELS = 1 // feel free to change this
const FINAL_SAMPLE_RATE = 44100 // feel free to change this
const FINAL_SAMPLE_RATE_CONVERSION = SAMPLE_RATE / FINAL_SAMPLE_RATE / CHANNELS
const SAMPLES = SECONDS * FINAL_SAMPLE_RATE // feel free to change this
const PRODUCT = SAMPLES * CHANNELS
BUFFER_SIZE = LIGHTNING_MODE ? PRODUCT : BUFFER_SIZE
const TYPE_BYTEBEAT = 0
const TYPE_SIGNED_BYTEBEAT = 1
const TYPE_FLOATBEAT = 2
const SELECTED_TYPE = TYPE_BYTEBEAT // feel free to change this
// for bytebeat
var int = x => Math.floor(x)
var abs = Math.abs
var acos = Math.acos
var acosh = Math.acosh
var asin = Math.asin
var asinh = Math.asinh
var atan = Math.atan
var atanh = Math.atanh
var cbrt = Math.cbrt
var cos = Math.cos
var cosh = Math.cosh
var floor = Math.floor
var log = Math.log
var log2 = Math.log2
var max = Math.max
var min = Math.min
var PI = Math.PI
var pow = Math.pow
var random = Math.random
var sin = Math.sin
var sinh = Math.sinh
var sqrt = Math.sqrt
var tan = Math.tan
var tanh = Math.tanh
2023-12-29 19:55:56 +03:00
const generateAudio = t => {
return t&t>>8
}
let lastCorrectSample = 127 // FIXME: guessed value
2023-12-29 19:55:56 +03:00
const constrainValue = sample => {
if (isNaN(sample) || sample < 0)
sample = lastCorrectSample
else
lastCorrectSample = sample
2023-12-29 19:55:56 +03:00
switch (SELECTED_TYPE) {
case TYPE_BYTEBEAT:
return sample & 255
case TYPE_SIGNED_BYTEBEAT:
return ((sample + 127) & 255) - 128
case TYPE_FLOATBEAT:
return floor(sample * 127) + 127
}
}
const random_choice = choices => choices[Math.floor(Math.random() * choices.length)]
2024-01-01 01:42:17 +03:00
const randomFileNameAlphabet =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789_-"
2023-12-29 19:55:56 +03:00
const generateRandomFilePath = () => {
2023-12-29 19:55:56 +03:00
let res = tmpdir() + "/" + basename(__filename) + "_"
for (let i = 0; i < 64; i++)
res += random_choice(randomFileNameAlphabet)
return res
}
let t = 0
let filePath = generateRandomFilePath()
writeFileSync(filePath, Buffer.alloc(0))
2023-12-29 19:55:56 +03:00
// the loop of sequential file writing, created to ease load on RAM
const buffer_max = Math.floor((PRODUCT + (BUFFER_SIZE - 1)) / BUFFER_SIZE),
needTwoBuffers = buffer_max > 1, needSingleBuffer = !needTwoBuffers
2023-12-29 19:55:56 +03:00
let audioData = new Uint8Array(needSingleBuffer ? PRODUCT : BUFFER_SIZE)
for (let seq = 0; seq < buffer_max; seq++) {
if (needTwoBuffers && (t + BUFFER_SIZE) >= PRODUCT) {
let calculatedSize = PRODUCT - t
audioData = new Uint8Array(calculatedSize)
}
for (let idx = 0; t < PRODUCT && idx < BUFFER_SIZE; idx++, t++) {
let sample = generateAudio(t * FINAL_SAMPLE_RATE_CONVERSION)
2023-12-29 19:55:56 +03:00
if (sample.constructor === Array)
sample.forEach((sample, index) => {
audioData[CHANNELS * idx + index] = constrainValue(sample)
2023-12-29 19:55:56 +03:00
})
else
audioData[idx] = constrainValue(sample)
2023-12-29 19:55:56 +03:00
}
appendFileSync(filePath, Buffer.from(audioData.buffer))
2023-12-29 19:55:56 +03:00
}
2024-01-01 01:42:17 +03:00
execSync(
`ffmpeg -f u8 -ar ${FINAL_SAMPLE_RATE} -ac ${CHANNELS} ` +
`-i ${filePath} output_${+new Date()}.wav`)
unlinkSync(filePath)