// // PCMPatchedTrackTests.m // Clock Signal // // Created by Thomas Harte on 17/12/2016. // Copyright © 2016 Thomas Harte. All rights reserved. // #import #include "PCMTrack.hpp" #include "PCMPatchedTrack.hpp" @interface PCMPatchedTrackTests : XCTestCase @end @implementation PCMPatchedTrackTests #pragma mark - Prebuilt tracks - (std::shared_ptr)togglingTrack { Storage::Disk::PCMSegment segment; segment.data = { 0xff, 0xff, 0xff, 0xff }; segment.number_of_bits = 32; return std::shared_ptr(new Storage::Disk::PCMTrack(segment)); } - (std::shared_ptr)patchableTogglingTrack { std::shared_ptr track = self.togglingTrack; return std::shared_ptr(new Storage::Disk::PCMPatchedTrack(track)); } - (std::shared_ptr)fourSegmentPatchedTrack { std::shared_ptr patchableTrack = self.patchableTogglingTrack; Storage::Disk::PCMPatchedTrack *patchable = static_cast(patchableTrack.get()); for(int c = 0; c < 4; c++) { Storage::Disk::PCMSegment segment; segment.data = {0xff}; segment.number_of_bits = 8; segment.length_of_a_bit.length = 1; segment.length_of_a_bit.clock_rate = 32; patchable->add_segment(Storage::Time(c, 4), segment, false); } return patchableTrack; } #pragma mark - - (std::vector)eventsFromTrack:(std::shared_ptr)track { std::vector events; while(1) { events.push_back(track->get_next_event()); if(events.back().type == Storage::Disk::Track::Event::IndexHole) break; } return events; } - (Storage::Time)timeForEvents:(const std::vector &)events { Storage::Time result(0); for(auto event : events) { result += event.length; } return result; } - (void)patchTrack:(std::shared_ptr)track withSegment:(Storage::Disk::PCMSegment)segment atTime:(Storage::Time)time { Storage::Disk::PCMPatchedTrack *patchable = static_cast(track.get()); patchable->add_segment(time, segment, false); } #pragma mark - Repeating Asserts - (void)assertOneThirtyTwosForTrack:(std::shared_ptr)track { // Confirm that there are now flux transitions (just the first five will do) // located 1/32nd of a rotation apart. for(int c = 0; c < 5; c++) { Storage::Disk::Track::Event event = track->get_next_event(); XCTAssert( event.length == (c ? Storage::Time(1, 32) : Storage::Time(1, 64)), @"flux transitions should be 1/32nd of a track apart"); } } - (void)assertEvents:(const std::vector &)events hasEntries:(size_t)numberOfEntries withEntry:(size_t)entry ofLength:(Storage::Time)time { XCTAssert(events.size() == numberOfEntries, @"Should be %zu total events", numberOfEntries); XCTAssert(events[entry].length == time, @"Event %zu should have been %d/%d long, was %d/%d", entry, time.length, time.clock_rate, events[entry].length.length, events[entry].length.clock_rate); Storage::Time eventTime = [self timeForEvents:events]; XCTAssert(eventTime == Storage::Time(1), @"Total track length should be 1; was %d/%d", eventTime.length, eventTime.clock_rate); } #pragma mark - Unpatched tracks - (void)testUnpatchedRawTrack { [self assertOneThirtyTwosForTrack:self.togglingTrack]; } - (void)testUnpatchedTrack { [self assertOneThirtyTwosForTrack:self.patchableTogglingTrack]; } #pragma mark - Insertions affecting one existing segment - (void)testSingleSplice { std::shared_ptr patchableTrack = self.patchableTogglingTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(1, 32), 1, {0xff}) atTime:Storage::Time(3, 128)]; std::vector events = [self eventsFromTrack:patchableTrack]; Storage::Time total_length = [self timeForEvents:events]; XCTAssert(events.size() == 33, @"Should still be 33 total events"); XCTAssert(events[0].length == Storage::Time(1, 64), @"First event should be after 1/64 as usual"); XCTAssert(events[1].length == Storage::Time(3, 128), @"Second event should be 3/128 later"); // ... as it was inserted at 3/128 and runs at the same rate as the main data, so first inserted event is at 3/128+1/64-1/64 XCTAssert(events[2].length == Storage::Time(5, 128), @"Should still be 33 total events"); // 1/64 = 2/128 to exit the patch, plus 3/128 to get to the next event, having spliced in 1/128 ahead of the normal clock XCTAssert(total_length == Storage::Time(1), @"Total track length should still be 1"); } - (void)testLeftReplace { std::shared_ptr patchableTrack = self.patchableTogglingTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(1, 16), 8, {0x00}) atTime:Storage::Time(0)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:17 withEntry:0 ofLength:Storage::Time(33, 64)]; } - (void)testRightReplace { std::shared_ptr patchableTrack = self.patchableTogglingTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(1, 16), 8, {0x00}) atTime:Storage::Time(1, 2)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:17 withEntry:16 ofLength:Storage::Time(33, 64)]; } #pragma mark - Insertions affecting three existing segments - (void)testMultiSegmentTrack { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:33 withEntry:4 ofLength:Storage::Time(1, 32)]; } - (void)testMultiTrimBothSideReplace { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(1, 16), 8, {0x00}) atTime:Storage::Time(1, 8)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:17 withEntry:4 ofLength:Storage::Time(17, 32)]; } - (void)testMultiTrimRightReplace { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(3, 8), 1, {0x00}) atTime:Storage::Time(1, 8)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:21 withEntry:4 ofLength:Storage::Time(13, 32)]; } - (void)testMultiTrimLeftReplace { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(3, 8), 1, {0x00}) atTime:Storage::Time(1, 4)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:21 withEntry:8 ofLength:Storage::Time(13, 32)]; } #pragma mark - Insertions affecting two existing segments - (void)testTwoSegmentOverlap { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(1, 32), 8, {0x00}) atTime:Storage::Time(1, 8)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:25 withEntry:4 ofLength:Storage::Time(9, 32)]; } - (void)testTwoSegmentRightReplace { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(3, 8), 1, {0x00}) atTime:Storage::Time(1, 8)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:21 withEntry:4 ofLength:Storage::Time(13, 32)]; } - (void)testTwoSegmentLeftReplace { std::shared_ptr patchableTrack = self.fourSegmentPatchedTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(3, 8), 1, {0x00}) atTime:Storage::Time(0)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:21 withEntry:0 ofLength:Storage::Time(25, 64)]; } #pragma mark - Wrapping segment - (void)testWrappingSegment { std::shared_ptr patchableTrack = self.patchableTogglingTrack; [self patchTrack:patchableTrack withSegment:Storage::Disk::PCMSegment(Storage::Time(5, 2), 1, {0x00}) atTime:Storage::Time(0)]; [self assertEvents:[self eventsFromTrack:patchableTrack] hasEntries:1 withEntry:0 ofLength:Storage::Time(1, 1)]; } @end