nickjillings@1392: /** nickjillings@1392: * loundess.js nickjillings@1392: * Loudness module for the Web Audio Evaluation Toolbox nickjillings@1392: * Allows for automatic calculation of loudness of Web Audio API Buffer objects, nickjillings@1392: * return gain values to correct for a target loudness or match loudness between nickjillings@1392: * multiple objects nickjillings@1392: */ nickjillings@1392: nickjillings@1392: var interval_cal_loudness_event = null; nickjillings@1392: nickjillings@1392: function calculateLoudness(buffer, timescale, target, offlineContext) nickjillings@1392: { nickjillings@1392: // This function returns the EBU R 128 specification loudness model and sets the linear gain required to match -23 LUFS nickjillings@1392: // buffer -> Web Audio API Buffer object nickjillings@1392: // timescale -> M or Momentary (returns Array), S or Short (returns Array), nickjillings@1392: // I or Integrated (default, returns number) nickjillings@1392: // target -> default is -23 LUFS but can be any LUFS measurement. nickjillings@1392: nickjillings@1392: if (buffer == undefined) nickjillings@1392: { nickjillings@1392: return 0; nickjillings@1392: } nickjillings@1392: if (timescale == undefined) nickjillings@1392: { nickjillings@1392: timescale = "I"; nickjillings@1392: } nickjillings@1392: if (target == undefined) nickjillings@1392: { nickjillings@1392: target = -23; nickjillings@1392: } nickjillings@1392: if (offlineContext == undefined) nickjillings@1392: { nickjillings@1392: offlineContext = new OfflineAudioContext(buffer.numberOfChannels, buffer.length, buffer.sampleRate); nickjillings@1392: } nickjillings@1392: // Create the required filters nickjillings@1392: var KFilter = offlineContext.createBiquadFilter(); nickjillings@1392: KFilter.type = "highshelf"; nickjillings@1392: KFilter.gain.value = 4; nickjillings@1392: KFilter.frequency.value = 1480; nickjillings@1392: nickjillings@1392: var HPFilter = offlineContext.createBiquadFilter(); nickjillings@1392: HPFilter.type = "highpass"; nickjillings@1392: HPFilter.Q.value = 0.707; nickjillings@1392: HPFilter.frequency.value = 60; nickjillings@1392: // copy Data into the process buffer nickjillings@1392: var processSource = offlineContext.createBufferSource(); nickjillings@1392: processSource.buffer = buffer; nickjillings@1392: nickjillings@1392: processSource.connect(KFilter); nickjillings@1392: KFilter.connect(HPFilter); nickjillings@1392: HPFilter.connect(offlineContext.destination); nickjillings@1392: processSource.start(); nickjillings@1392: offlineContext.startRendering().then(function(renderedBuffer) { nickjillings@1392: // Have the renderedBuffer information, now continue processing nickjillings@1392: switch(timescale) nickjillings@1392: { nickjillings@1392: case "I": nickjillings@1392: var blockEnergy = calculateProcessedLoudness(renderedBuffer, 400, 0.75); nickjillings@1392: // Apply the absolute gate nickjillings@1392: var loudness = calculateLoudnessFromChannelBlocks(blockEnergy); nickjillings@1392: var absgatedEnergy = new Array(blockEnergy.length); nickjillings@1392: for (var c=0; c= -70) nickjillings@1392: { nickjillings@1392: for (var c=0; c= relGateLevel) nickjillings@1392: { nickjillings@1392: for (var c=0; c= 4) {G = 1.41;} nickjillings@1392: sigma += blockEnergy[channel][i]*G; nickjillings@1392: } nickjillings@1392: loudness[i] = -0.691 + 10*Math.log10(sigma); nickjillings@1392: } nickjillings@1392: return loudness; nickjillings@1392: } nickjillings@1392: function calculateOverallLoudnessFromChannelBlocks(blockEnergy) nickjillings@1392: { nickjillings@1392: // Loudness nickjillings@1392: var summation = 0; nickjillings@1392: for (var channel = 0; channel < blockEnergy.length; channel++) nickjillings@1392: { nickjillings@1392: var G = 1.0; nickjillings@1392: if (channel >= 4) {G = 1.41;} nickjillings@1392: var sigma = 0; nickjillings@1392: for (var i=0; i