giuliomoro@1304: /** giuliomoro@1304: * loundess.js giuliomoro@1304: * Loudness module for the Web Audio Evaluation Toolbox giuliomoro@1304: * Allows for automatic calculation of loudness of Web Audio API Buffer objects, giuliomoro@1304: * return gain values to correct for a target loudness or match loudness between giuliomoro@1304: * multiple objects giuliomoro@1304: */ giuliomoro@1304: giuliomoro@1304: var interval_cal_loudness_event = null; giuliomoro@1304: giuliomoro@1304: if (typeof OfflineAudioContext == "undefined"){ giuliomoro@1304: var OfflineAudioContext = webkitOfflineAudioContext; giuliomoro@1304: } giuliomoro@1304: giuliomoro@1304: function calculateLoudness(buffer, timescale, target, offlineContext) giuliomoro@1304: { giuliomoro@1304: // This function returns the EBU R 128 specification loudness model and sets the linear gain required to match -23 LUFS giuliomoro@1304: // buffer -> Web Audio API Buffer object giuliomoro@1304: // timescale -> M or Momentary (returns Array), S or Short (returns Array), giuliomoro@1304: // I or Integrated (default, returns number) giuliomoro@1304: // target -> default is -23 LUFS but can be any LUFS measurement. giuliomoro@1304: giuliomoro@1304: if (buffer == undefined) giuliomoro@1304: { giuliomoro@1304: return 0; giuliomoro@1304: } giuliomoro@1304: if (timescale == undefined) giuliomoro@1304: { giuliomoro@1304: timescale = "I"; giuliomoro@1304: } giuliomoro@1304: if (target == undefined) giuliomoro@1304: { giuliomoro@1304: target = -23; giuliomoro@1304: } giuliomoro@1304: if (offlineContext == undefined) giuliomoro@1304: { giuliomoro@1304: offlineContext = new OfflineAudioContext(audioContext.destination.channelCount, buffer.buffer.duration*audioContext.sampleRate, audioContext.sampleRate); giuliomoro@1304: } giuliomoro@1304: // Create the required filters giuliomoro@1304: var KFilter = offlineContext.createBiquadFilter(); giuliomoro@1304: KFilter.type = "highshelf"; giuliomoro@1304: KFilter.gain.value = 4; giuliomoro@1304: KFilter.frequency.value = 1500; giuliomoro@1304: giuliomoro@1304: var HPFilter = offlineContext.createBiquadFilter(); giuliomoro@1304: HPFilter.type = "highpass"; giuliomoro@1304: HPFilter.Q.value = 0.5; giuliomoro@1304: HPFilter.frequency.value = 38; giuliomoro@1304: // copy Data into the process buffer giuliomoro@1304: var processSource = offlineContext.createBufferSource(); giuliomoro@1304: processSource.buffer = buffer.buffer; giuliomoro@1304: giuliomoro@1304: processSource.connect(KFilter); giuliomoro@1304: KFilter.connect(HPFilter); giuliomoro@1304: HPFilter.connect(offlineContext.destination); giuliomoro@1304: processSource.start(); giuliomoro@1304: offlineContext.oncomplete = function(renderedBuffer) { giuliomoro@1304: // Have the renderedBuffer information, now continue processing giuliomoro@1304: if (typeof renderedBuffer.renderedBuffer == 'object') { giuliomoro@1304: renderedBuffer = renderedBuffer.renderedBuffer; giuliomoro@1304: } giuliomoro@1304: switch(timescale) giuliomoro@1304: { giuliomoro@1304: case "I": giuliomoro@1304: // Calculate the Mean Squared of a signal giuliomoro@1304: var MS = calculateMeanSquared(renderedBuffer,0.4,0.75); giuliomoro@1304: // Calculate the Loudness of each block giuliomoro@1304: var MSL = calculateLoudnessFromBlocks(MS); giuliomoro@1304: // Get blocks from Absolute Gate giuliomoro@1304: var LK = loudnessGate(MSL,MS,-70); giuliomoro@1304: // Calculate Loudness giuliomoro@1304: var LK_gate = loudnessOfBlocks(LK); giuliomoro@1304: // Get blocks from Relative Gate giuliomoro@1304: var RK = loudnessGate(MSL,MS,LK_gate-10); giuliomoro@1304: var RK_gate = loudnessOfBlocks(RK); giuliomoro@1304: buffer.buffer.lufs = RK_gate; giuliomoro@1304: } giuliomoro@1304: buffer.ready(); giuliomoro@1304: }; giuliomoro@1304: offlineContext.startRendering(); giuliomoro@1304: } giuliomoro@1304: giuliomoro@1304: function calculateMeanSquared(buffer,frame_dur,frame_overlap) giuliomoro@1304: { giuliomoro@1304: frame_size = Math.floor(buffer.sampleRate*frame_dur); giuliomoro@1304: step_size = Math.floor(frame_size*(1.0-frame_overlap)); giuliomoro@1304: num_frames = Math.floor((buffer.length-frame_size)/step_size); giuliomoro@1304: giuliomoro@1304: MS = Array(buffer.numberOfChannels); giuliomoro@1304: for (var c=0; c= 3){G = 1.41;} giuliomoro@1304: sum += blocks[c][n]*G; giuliomoro@1304: } giuliomoro@1304: MSL[n] = -0.691 + 10*Math.log10(sum); giuliomoro@1304: } giuliomoro@1304: return MSL; giuliomoro@1304: } giuliomoro@1304: giuliomoro@1304: function loudnessGate(blocks,source,threshold) giuliomoro@1304: { giuliomoro@1304: var num_frames = source[0].length; giuliomoro@1304: var num_channels = source.length; giuliomoro@1304: var LK = Array(num_channels); giuliomoro@1304: for (var c=0; c threshold) giuliomoro@1304: { giuliomoro@1304: for (var c=0; c= 3){G = 1.41;} giuliomoro@1304: sum += blocks[c][n]*G; giuliomoro@1304: } giuliomoro@1304: sum /= num_frames; giuliomoro@1304: loudness += sum; giuliomoro@1304: } giuliomoro@1304: loudness = -0.691 + 10 * Math.log10(loudness); giuliomoro@1304: return loudness; giuliomoro@1304: }