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