1 /**
2 Safe, flexible, audio buffer RAII structure.
3 
4 Copyright: Copyright Guillaume Piolat 2021.
5 License:   $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0)
6 */
7 module dplug.audio.audiobuffer;
8 
9 import core.bitop: bsf;
10 import core.stdc.string;
11 import dplug.core.math;
12 import inteli.emmintrin;
13 
14 // A word of warning:
15 // Do not try to use "inout" in this file.
16 // this is the path to misery.
17 
18 nothrow:
19 @nogc:
20 @safe:
21 
22 /// Allocate a new `AudioBuffer` with given `frames` and `channels`.
23 /// Its data is _not_ initialized.
24 AudioBuffer!T audioBufferAlloc(T)(int channels, int frames, int alignment = 1)
25 {
26     AudioBuffer!T buf;
27     buf.resize(channels, frames, alignment);
28     return buf;
29 }
30 
31 /// Allocate a new `AudioBuffer` with given `frames` and `channels`.
32 /// Its data is initialized to zeroes.
33 AudioBuffer!T audioBufferAllocZeroed(T)(int channels, int frames, int alignment = 1)
34 {
35     AudioBuffer!T buf;
36     buf.resize(channels, frames, alignment);
37     buf.fillWithZeroes();
38     return buf;
39 }
40 
41 /// Create a `AudioBuffer` by reusing existing data. Hence no format conversion happens.
42 AudioBuffer!T audioBufferFromData(T)(int channels, int frames, T** inData) @system
43 {
44     AudioBuffer!T buf;
45     buf.initWithData(channels, frames, inData);
46     return buf;
47 }
48 ///ditto
49 const(AudioBuffer!T) audioBufferFromData(T)(int channels, int frames, const(T*)* inData) @system
50 {    
51     int frameStart = 0;
52     ubyte alignment = 1;
53     ubyte flags = 0;
54     return const(AudioBuffer!T)(channels, frames, inData, frameStart, alignment, flags);
55 }
56 
57 /*
58 ///ditto
59 const(AudioBuffer!T) audioBufferCreateSubBuffer(T)(ref const(AudioBuffer!T) parent, int frameStart, int frameEnd) @trusted
60 {
61     assert(frameStart >= 0);
62     assert(frameStart <= frameEnd);
63     assert(frameEnd <= parent.frames());
64     ubyte alignment = 1;
65     int channels = parent.channels();
66     int frames = frameEnd - frameStart;
67 
68     const(T*)* data = parent.getChannelsPointers();
69     bool parentHasZeroFlag = parent.hasZeroFlag();
70 
71     return const(AudioBuffer!T)(channels, frames, data, frameStart,
72                                 alignment, parentHasZeroFlag ? AudioBuffer!T.Flags.isZero : 0);
73 }
74 */
75 
76 /// Duplicate an `AudioBuffer` with an own allocation, make it mutable.
77 AudioBuffer!T audioBufferDup(T)(ref const(AudioBuffer!T) buf, int alignment = 1)
78 {
79     AudioBuffer!T b;
80     b.resize(buf.channels(), buf.frames(), alignment);
81     b.copyFrom(buf);
82     return b;
83 }
84 
85 
86 /// An `AudioBuffer` is a multi-channel buffer, with defined length, to act as storage 
87 /// of audio samples of type `T`.
88 /// It is passed around by DSP algorithms.
89 /// Data is store deinterleaved.
90 struct AudioBuffer(T)
91 {
92 public:
93 nothrow:
94 @nogc:
95 @safe:
96 
97     /// Change the size (`channels` and `frames`) of the underlying store.
98     /// Data is left uninitialized.
99     /// Typically you would reuse an `AudioBuffer` if you want to reuse the allocation.
100     /// When the same size is requested, the same allocation is reused (unless alignment is changed).
101     void resize(int channels, int frames, int alignment = 1)
102     {
103         resizeDiscard(channels, frames, alignment);
104 
105         // Debug: fill with NaNs, this will make non-initialization problem very explicit.
106         debug
107         {
108             fillWithValue(T.nan);
109         }
110     }
111 
112     /// Dispose the previous content, if any.
113     /// Allocate a new `AudioBuffer` with given `frames` and `channels`. This step can reuse an existing owned allocation.
114     void initWithData(int channels, int frames, T** inData) @system
115     {
116         assert(channels <= maxPossibleChannels);
117 
118         // Release own memory if any.
119         cleanUpData();
120 
121         _channels = channels;
122         _frames = frames; 
123         _alignment = 1;
124         _flags = Flags.hasOtherMutableReference;
125 
126         for (int n = 0; n < channels; ++n)
127         {
128             _channelPointers[n] = inData[n];
129         }
130     }
131 
132     ~this()
133     {
134         cleanUp();
135     }
136 
137     /// Return number of channels in this buffer.
138     int channels() const
139     {
140         return _channels;
141     }
142 
143     /// Return number of frames in this buffer.
144     int frames() const
145     {
146         return _frames;
147     }
148 
149     /// Return alignment of the sample storage.
150     int alignment() const
151     {
152         return _alignment;
153     }
154 
155 
156     @disable this(this);
157 
158     /// Recompute the status of the zero flag.
159     /// Otherwise, asking for a mutable pointer into the zero data will clear this flag.
160     /// This is useful if you've written in a buffer and want to optimize downstream.
161     void recomputeZeroFlag()
162     {
163         if (computeIsBufferSilent())
164             setZeroFlag();
165         else
166         {
167             // Normally the zero flag is already unset.
168             // is is a logical error if the zero flag is set when it can be non-zero
169             assert(!hasZeroFlag);
170         }
171     }
172 
173     /// Returns: true if the buffer is all zeroes.
174     bool isSilent() const
175     {
176         if (hasZeroFlag())
177         {
178             if (isIsolated())
179                 return true;
180             else
181                 return computeIsBufferSilent();
182         }
183         else
184             return computeIsBufferSilent();
185     }
186 
187     /// Returns: `true` is the data is only pointed to by this `AudioBuffer`.
188     bool isIsolated() const
189     {
190         return (_flags & Flags.hasOtherMutableReference)== 0;
191     }
192 
193     // This break the isolated flag manually, in case you want to be able 
194     // to use the zero flag regardless, at your own risk.
195     void assumeIsolated()
196     {
197         clearHasOtherMutableReferenceFlag();
198     }
199 
200     /// Returns: `true` is the buffer own its pointed audio data.
201     bool hasOwnership() const
202     {
203         return (_flags & Flags.hasOwnership) != 0;
204     }
205 
206     // <data-access>
207 
208     /// Get pointer of a given channel.
209     T* getChannelPointer(int channel)
210     {
211         clearZeroFlag();
212         return _channelPointers[channel];
213     }
214 
215     /// Get const pointer of a given channel.
216     const(T)* getChannelPointer(int channel) const
217     {
218         return _channelPointers[channel];
219     }
220 
221     /// Get immutable pointer of a given channel.
222     immutable(T)* getChannelPointer(int channel) immutable
223     {
224         return _channelPointers[channel];
225     }
226 
227     /// Get channel pointers.
228     T** getChannelsPointers() return @trusted
229     {
230         clearZeroFlag();
231         return _channelPointers.ptr;
232     }
233 
234     /// Get const channel pointers.
235     const(T*)* getChannelsPointers() return @trusted const
236     {
237         return _channelPointers.ptr;
238     }
239 
240     /// Get immutable channel pointers.
241     immutable(T*)* getChannelsPointers() return @trusted immutable
242     {
243         return _channelPointers.ptr;
244     }
245 
246     /// Get slice of a given channel.
247     T[] getChannel(int channel) @trusted
248     {
249         clearZeroFlag();
250         return _channelPointers[channel][0.._frames];
251     }
252     ///ditto
253     const(T)[] getChannel(int channel) const @trusted
254     {
255         return _channelPointers[channel][0.._frames];
256     }
257     ///ditto
258     immutable(T)[] getChannel(int channel) immutable @trusted
259     {
260         return _channelPointers[channel][0.._frames];
261     }
262     ///ditto
263     inout(T)[] opIndex(int channel) inout @trusted
264     {
265         return _channelPointers[channel][0.._frames];
266     }
267 
268     // </data-access>
269 
270     // <opIndex>
271 
272     /// Create an AudioBuffer that is a ref to the same data.
273     AudioBuffer opIndex()
274     {
275         return sliceFrames(0, _frames);
276     }
277     ///ditto
278     const(AudioBuffer) opIndex() const
279     {
280         return sliceFrames(0, _frames);
281     }
282 
283     /// Index a single sample.
284     ref inout(T) opIndex(int channel, int frame) inout @trusted
285     { 
286         return _channelPointers[channel][frame];
287     }
288 
289     /// Slice with a sub-range of channels.
290     AudioBuffer opIndex(int[2] chan)
291     {
292         return sliceChannels(chan[0], chan[1]);
293     } 
294     ///ditto
295     const(AudioBuffer) opIndex(int[2] chan) const
296     {
297         return sliceChannels(chan[0], chan[1]);
298     }
299 
300     /// Slice across channels and temporally.
301     /// Take a sub-range of channels, and a sub-range of frames.
302     AudioBuffer opIndex(int[2] chan, int[2] framesBounds)
303     {
304         return sliceChannels(chan[0], chan[1]).sliceFrames(framesBounds[0], framesBounds[1]);
305     } 
306     ///ditto
307     const(AudioBuffer) opIndex(int[2] chan, int[2] framesBounds) const
308     {
309         return sliceChannels(chan[0], chan[1]).sliceFrames(framesBounds[0], framesBounds[1]);
310     }
311 
312     // </op-index>
313 
314     // <slice>
315 
316     int opDollar(size_t dim : 0)() pure const 
317     { 
318         return _channels; 
319     }
320 
321     int opDollar(size_t dim : 1)() pure const
322     { 
323         return _frames; 
324     }
325 
326     /// Select only a slice of channels from `AudioBuffer`.
327     /// Params:
328     ///    frameStart offset in the buffer. Must be >= 0 and <= `frameEnd`.
329     ///    frameEnd offset in the buffer. Cannot be larger than the parent size.
330     AudioBuffer sliceChannels(int channelStart, int channelEnd) @trusted
331     {
332         assert(channelStart >= 0);
333         assert(channelStart <= channelEnd);
334         assert(channelEnd <= this.channels());
335         int channelSub = channelEnd - channelStart;
336         T** data = this.getChannelsPointers();
337 
338         ubyte flags = Flags.hasOtherMutableReference;
339         if (this.hasZeroFlag())
340             flags |= Flags.isZero;
341 
342         // Because this is a mutable reference, both the parent and the result
343         // get the "has another mutable ref" flag.
344         this.setHasOtherMutableReferenceFlag();
345 
346         return AudioBuffer!T(channelSub, 
347                              this.frames(), 
348                              data + channelStart,
349                              0, 
350                              _alignment,
351                              flags);
352     }
353     //ditto
354     const(AudioBuffer) sliceChannels(int channelStart, int channelEnd) const @trusted
355     {
356         assert(channelStart >= 0);
357         assert(channelStart <= channelEnd);
358         assert(channelEnd <= this.channels());
359         int channelSub = channelEnd - channelStart;
360         const(T*)* data = this.getChannelsPointers();
361 
362         ubyte flags = 0;
363         if (this.hasZeroFlag())
364             flags |= Flags.isZero;
365 
366         return AudioBuffer!T(channelSub, 
367                              this.frames(), 
368                              data + channelStart,
369                              0, 
370                              _alignment,
371                              flags);
372     }
373     ///ditto
374     int[2] opSlice(size_t dim)(int start, int end) const
375     {
376         return [start, end];
377     }
378 
379     /// Create a `AudioBuffer` derivated from another buffer.
380     /// Params:
381     ///    frameStart offset in the buffer. Must be >= 0 and <= `frameEnd`.
382     ///    frameEnd offset in the buffer. Cannot be larger than the parent size.
383     AudioBuffer sliceFrames(int frameStart, int frameEnd) @trusted
384     {
385         assert(frameStart >= 0);
386         assert(frameStart <= frameEnd);
387         assert(frameEnd <= this.frames());
388         ubyte alignment = childAlignment(_alignment, T.sizeof, frameStart);
389         int channels = this.channels();
390         int framesSub = frameEnd - frameStart;
391 
392         T** data = this.getChannelsPointers();
393 
394         ubyte flags = Flags.hasOtherMutableReference;
395         if (this.hasZeroFlag())
396             flags |= Flags.isZero;
397 
398         // Because this is a mutable reference, both the parent and the result
399         // get the "has another mutable ref" flag.
400         this.setHasOtherMutableReferenceFlag();
401 
402         return AudioBuffer!T(channels, 
403                              framesSub, 
404                              data,
405                              frameStart, 
406                              alignment,
407                              flags);
408     }
409     ///ditto
410     const(AudioBuffer) sliceFrames(int frameStart, int frameEnd) const @trusted
411     {
412         assert(frameStart >= 0);
413         assert(frameStart <= frameEnd);
414         assert(frameEnd <= this.frames());
415         ubyte alignment = childAlignment(_alignment, T.sizeof, frameStart);
416         int channels = this.channels();
417         int framesSub = frameEnd - frameStart;
418 
419         const(T*)* data = this.getChannelsPointers();
420 
421         ubyte flags = 0;
422         if (this.hasZeroFlag())
423             flags |= Flags.isZero;
424 
425         return AudioBuffer!T(channels, 
426                              framesSub, 
427                              data,
428                              frameStart, 
429                              alignment,
430                              flags);
431     }
432 
433     // </slice>
434 
435     // <copy>
436 
437     /// Copy samples from `source` to `dest`.
438     /// Number of `frames` and `channels` must match.
439     void copyFrom(ref const(AudioBuffer) source) @trusted
440     {
441         assert(_frames == source.frames());
442         assert(_channels == source.channels());
443 
444         size_t bytesForOneChannel = T.sizeof * _frames;
445         for (int chan = 0; chan < _channels; ++chan)
446         {
447             memmove(_channelPointers[chan], source._channelPointers[chan], bytesForOneChannel);
448         }
449         if (source.hasZeroFlag)
450             setZeroFlag();
451     }
452 
453     // </copy>
454 
455 
456     // <filling the buffer>
457 
458     /// Fill the buffer with zeroes.
459     void fillWithZeroes() @trusted
460     {
461         size_t bytesForOneChannel = T.sizeof * _frames;
462         for (int chan = 0; chan < _channels; ++chan)
463         {
464             memset(_channelPointers[chan], 0, bytesForOneChannel);
465         }
466         setZeroFlag();
467     }
468 
469     /// Fill the buffer with a single value.
470     /// Warning: the buffer must be in `fp32` format.
471     void fillWithValue(T value) @trusted
472     {
473         if (value == 0)
474             return fillWithZeroes(); // Note: this turns -0.0 into +0.0
475 
476         for (int chan = 0; chan < _channels; ++chan)
477         {
478             T* p = getChannelPointer(chan);
479             p[0.._frames] = value;
480         }
481         assert(!hasZeroFlag);
482     }
483 
484     // </filling the buffer>
485 
486     // <buffer splitting>
487 
488     /// Return an input range that returns several subbuffers that covers the 
489     /// parent buffer, each with length not larger than `maxFrames`.
490     auto chunkBy(int maxFrames)
491     {
492         static struct AudioBufferRange
493         {
494             AudioBuffer buf;
495             int offset = 0;
496             int maxFrames;
497             int totalFrames;
498 
499             AudioBuffer front()
500             {
501                 int end = offset + maxFrames;
502                 if (end > totalFrames)
503                     end = totalFrames;
504                 AudioBuffer res = buf.sliceFrames(offset, end);
505                 return res;
506             }
507 
508             void popFront()
509             {
510                 offset += maxFrames;
511             }
512 
513             bool empty()
514             {
515                 return offset >= totalFrames;
516             }
517         }
518         return AudioBufferRange( sliceFrames(0, frames()), 0, maxFrames, frames() );
519     }
520     ///ditto
521     auto chunkBy(int maxFrames) const
522     {
523         static struct AudioBufferRange
524         {
525             const(AudioBuffer) buf;
526             int offset;
527             int maxFrames;
528             int totalFrames;
529 
530             const(AudioBuffer) front()
531             {
532                 int end = offset + maxFrames;
533                 if (end > totalFrames)
534                     end = totalFrames;
535                 const(AudioBuffer) res = buf.sliceFrames(offset, end);
536                 return res;
537             }
538 
539             void popFront()
540             {
541                 offset += maxFrames;
542             }
543 
544             bool empty()
545             {
546                 return offset >= totalFrames;
547             }
548         }
549         return AudioBufferRange( sliceFrames(0, frames()), 0, maxFrames, frames() );
550     }
551 
552     // </buffer splitting>
553 
554 private:
555 
556     /// Internal flags
557     enum Flags : ubyte
558     {
559         /// Zero flag.
560         isZero = 1,
561 
562         /// Owner flag.
563         hasOwnership = 2, 
564 
565         /// Growable flag.
566         //isGrowable = 4
567 
568         /// Data is pointed to, mutably, by other things than this `AudioBuffer`.
569         hasOtherMutableReference = 8,
570     }
571 
572     // TODO: lift that limitation
573     enum maxPossibleChannels = 8;
574 
575     // Pointers to beginning of every channel.
576     T*[maxPossibleChannels] _channelPointers;
577     
578     // Pointer to start of data.
579     // If memory is _owned_, then it is allocated with `_mm_realloc_discard`/`_mm_free`
580     void* _data     = null;
581 
582     // Number of frames in the buffer.
583     int _frames;
584 
585     // Number of channels in the buffer.
586     int _channels;
587 
588     // Various flags.
589     ubyte _flags;
590 
591     // Current allocation alignment.
592     ubyte _alignment = 0; // current alignment, 0 means "unassigned" 
593 
594     // Test if the zero flag is set.
595     // Returns: `true` is the buffer has the `zeroFlag` set.
596     // Warning: that when this flag isn't set, the buffer could still contains only zeroes.
597     //          If you want to test for zeroes, use `isSilent` instead.
598     bool hasZeroFlag() const
599     {
600         return (_flags & Flags.isZero) != 0;
601     }    
602 
603     void clearZeroFlag()
604     {
605         _flags &= ~cast(int)Flags.isZero;
606     }
607 
608     void setZeroFlag()
609     {
610         _flags |= Flags.isZero;
611     }
612 
613     void setHasOtherMutableReferenceFlag()
614     {
615         _flags |= Flags.hasOtherMutableReference;
616     }
617 
618     void clearHasOtherMutableReferenceFlag()
619     {
620         _flags &= ~cast(int)Flags.hasOtherMutableReference;
621     }
622 
623     // Private constructor, the only way to create const/immutable object.
624     this(int channels, 
625          int frames, 
626          const(T*)* inData, 
627          int offsetFrames, // point further in the input
628          ubyte alignment, 
629          ubyte flags) @system 
630     {
631         assert(offsetFrames >= 0);
632         assert(channels <= maxPossibleChannels);
633         _channels = channels;
634         _frames = frames; 
635         _alignment = alignment;
636         _flags = flags;
637 
638         for (int n = 0; n < channels; ++n)
639         {
640             _channelPointers[n] = cast(T*)(inData[n]) + offsetFrames; // const_cast here
641         }
642     }
643 
644     void resizeDiscard(int channels, int frames, int alignment) @trusted
645     {
646         assert(channels >= 0 && frames >= 0);
647         assert(alignment >= 1 && alignment <= 128);
648         assert(isPowerOfTwo(alignment));
649         assert(channels <= maxPossibleChannels); // TODO allocate to support arbitrary channel count.
650 
651         if (_alignment != 0 && _alignment != alignment)
652         {
653             // Can't keep the allocation if the alignment changes.
654             cleanUpData();
655         }
656 
657         _channels = channels;
658         _frames = frames; 
659         _alignment = cast(ubyte)alignment;
660 
661         size_t bytesForOneChannel = T.sizeof * frames;
662         bytesForOneChannel = nextMultipleOf(bytesForOneChannel, alignment);
663         
664         size_t bytesTotal = bytesForOneChannel * channels;
665         if (bytesTotal == 0) 
666             bytesTotal = 1; // so that zero length or zero-channel buffers still kinda work.
667 
668         _data = _mm_realloc_discard(_data, bytesTotal, alignment);
669         _flags = Flags.hasOwnership;
670 
671         for (int n = 0; n < _channels; ++n)
672         {
673             ubyte* p = (cast(ubyte*)_data) + bytesForOneChannel * n;
674             _channelPointers[n] = cast(T*)p;
675         }
676     }
677 
678     void cleanUp()
679     {
680         cleanUpData();
681     }
682 
683     void cleanUpData()
684     {
685         if (hasOwnership())
686         {
687             if (_data !is null)
688             {
689                 _mm_free(_data);
690                 _data = null;
691             }
692         }
693 
694         // Note: doesn't loose the ownership flag, because morally this AudioBuffer is still 
695         // the kind of AudioBuffer that owns its data, it just has no data right now.
696     }
697 
698     // Returns: true if the whole buffer is filled with 0 (or -0 for floating-point) 
699     // Do not expose this API as it isn't clear how fast it is.
700     bool computeIsBufferSilent() const nothrow @nogc @trusted
701     {
702         for (int channel = 0; channel < _channels; ++channel)
703         {
704             const(T)* samples = getChannelPointer(channel);
705             for (int n = 0; n < _frames; ++n)
706             {
707                 if (samples[n] != 0)
708                     return false;
709             }
710         }
711         return true;
712     }
713 }
714 
715 private:
716 
717 // Compute largest possible byte alignment for a sub-buffer.
718 ubyte childAlignment(ubyte parentAlignment, size_t itemSize, int frameStart) pure
719 {
720     assert(parentAlignment >= 1 && parentAlignment <= 128);
721 
722     // For reference, this is the alignment for T == float:
723     //
724     // float(4 bytes)|  0   |  1  |  2   |  3  |   4 |
725     // ----------------------------------------------|
726     // parent 1      |  1   |  1  |  1   |  1  |   1 |
727     // parent 2      |  2   |  2  |  2   |  2  |   2 |
728     // parent 4      |  4   |  4  |  4   |  4  |   4 |
729     // parent 8      |  8   |  4  |  8   |  4  |   8 |
730     // parent 16     |  16  |  4  |  8   |  4  |  16 |
731     // parent 32     |  32  |  4  |  8   |  4  |  16 |
732 
733     size_t offset = frameStart * itemSize;
734     // how many zero bits there are in LSB?
735     if (offset == 0)
736         return parentAlignment;
737     int zeroBits = bsf(offset);
738     if (zeroBits > 7) 
739         zeroBits = 7; // do not exceed 128
740     int a = (1 << zeroBits);
741     if (a > parentAlignment)
742         a = parentAlignment;
743     return cast(ubyte)a;
744 }
745 unittest
746 {
747     assert( childAlignment(1, 8, 8) == 1 );        // do not exceed aprent align
748     assert( childAlignment(16, 4, 2) == 8 );
749     assert( childAlignment(16, 8, 0) == 16 );
750     assert( childAlignment(16, 8, 1024) == 16 );
751     assert( childAlignment(16, 4, 1) == 4 );
752     assert( childAlignment(128, 4, 1024) == 128 ); // do not exceed 128
753 }
754 
755 // How zero flag works:
756 @trusted unittest 
757 {
758     AudioBuffer!double a = audioBufferAlloc!double(3, 1024, 16);
759     assert(!a.hasZeroFlag());
760     assert(!a.isSilent);
761     a.fillWithValue(0.0);
762 
763     // Newly created AudioBuffer with own memory is zeroed out.
764     assert(a.hasZeroFlag());
765     assert(a.isSilent);
766 
767     // Getting a mutable pointer make the zero flag disappear.
768     a.getChannelPointer(2)[1023] = 0.0;
769     assert(!a.hasZeroFlag());
770     assert(a.isSilent());
771 
772     // To set the zero flag, either recompute it (slow)
773     // or fill the buffer with zeroes.
774     a.recomputeZeroFlag();
775     assert(a.hasZeroFlag());
776     assert(a.isSilent());
777 }
778 
779 unittest 
780 {
781     // Buffer must reuse an existing allocation if the size/alignment is the same.
782     // Even in this case, the content is NOT preserved.
783     AudioBuffer!double b;
784     b.resize(1, 1024, 16);
785     double* chan0 = b.getChannelPointer(0);
786 
787     b.resize(1, 1024, 16);
788     assert(chan0 == b.getChannelPointer(0));
789 
790     // If the alignment changes though, can't reuse allocation.
791     b.resize(1, 1024, 128);
792     assert(b.channels() == 1);
793     assert(b.frames() == 1024);
794     b.resize(2, 1023, 128);
795 
796     b.fillWithValue(4.0);
797     double[] p = b.getChannel(1);
798     assert(p[1022] == 4.0);
799     assert(!b.hasZeroFlag());
800 }
801 
802 @trusted unittest
803 {
804     // const borrow preserve zero flag
805     int numChans = 8;
806     const(AudioBuffer!float) c = audioBufferAllocZeroed!float(numChans, 123);
807     assert(c.hasZeroFlag());
808     const(float*)* buffers = c.getChannelsPointers();
809     for (int chan = 0; chan < 8; ++chan)
810     {
811         assert(buffers[chan] == c.getChannelPointer(chan));
812     }
813     const(AudioBuffer!float) d = c.sliceFrames(10, c.frames());
814     assert(d.frames() == 123 - 10);
815     assert(d.hasZeroFlag());
816 
817     const(AudioBuffer!float) e = d[0..$, 0..24];
818     assert(e.channels() == d.channels());
819     assert(e.frames() == 24);
820 }
821 
822 @trusted unittest
823 {
824     // Mutable borrow doesn't preserve zero flag, and set hasOtherMutableReference flag
825     int numChans = 2;
826     AudioBuffer!double c = audioBufferAlloc!double(numChans, 14);
827     c.fillWithZeroes();
828     assert(c.hasZeroFlag());
829     assert(c.isIsolated());
830     
831     AudioBuffer!double d = c[0..$, 10 .. 14];
832     assert(!c.hasZeroFlag());
833     assert(!d.hasZeroFlag());
834     assert(!c.isIsolated);
835     assert(!d.isIsolated);
836 
837     // Fill right channel with 2.5f 
838     d.getChannel(1)[] = 2.5;
839     assert(c[1, 9] == 0.0);
840     c[1, 8] = -1.0;
841     assert(c[1, 8] == -1.0);    
842     
843     assert(c[1][10] == 2.5);
844 
845     // Mutable dup
846     AudioBuffer!double e = audioBufferDup(d);
847     assert(e.isIsolated);
848 }
849 
850 @trusted unittest
851 {   
852     // Create mutable buffer from mutable data.
853     {
854         float[128][2] data;
855         float*[2] pdata;
856         pdata[0] = data[0].ptr;
857         pdata[1] = data[1].ptr;
858         AudioBuffer!float b;
859         b.initWithData(2, 128, pdata.ptr);
860         assert(!b.isIsolated);
861 
862         // This break the isolated flag manually, in case you want to be able 
863         // to use the zero flag regardless, at your own risk.
864         b.assumeIsolated(); 
865         assert(b.isIsolated);
866     }
867 
868     // Create const buffer from const data.
869     {
870         float[128][2] data;
871         const(float*)[2] pdata = [ data[0].ptr, data[1].ptr];
872         const(AudioBuffer!float) b = audioBufferFromData(2, 128, pdata.ptr);
873         assert(b.isIsolated);
874     }
875 }
876 
877 unittest
878 {
879     // Chunked foreach
880     {
881         AudioBuffer!double whole = audioBufferAlloc!double(2, 323 + 1024, 16);
882         foreach(b; whole.chunkBy(1024))
883         {
884             assert(b.frames() <= 1024);
885             b.fillWithZeroes();
886 
887             AudioBuffer!double c = b[0..$];
888             assert(c.channels == whole.channels);
889             assert(c.alignment == whole.alignment); // inherited alignment correctly
890             assert(c.frames == b.frames);
891         }
892         assert(whole.computeIsBufferSilent());
893     }
894 
895     // Chunked const foreach
896     {
897         const(AudioBuffer!double) whole = audioBufferAllocZeroed!double(3, 2000);
898         foreach(b; whole.chunkBy(1024)) // split by frames
899         {
900             assert(b.isSilent);
901 
902             // Split by channels
903             const(AudioBuffer!double) left = b[0..1];
904             const(AudioBuffer!double) right = b.sliceChannels(1, 2);
905             assert(left.isSilent);
906             assert(b.frames == left.frames);
907             assert(left.channels == 1);
908             assert(right.isSilent);
909         }
910     }
911 }