summaryrefslogtreecommitdiff
path: root/SmartDeviceLink/private/SDLProtocolMessageAssembler.m
blob: e4a07541c52018ca0f4e4cc9e54a852bc3d6b95c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
//  SDLSmartDeviceLinkProtocolMessageAssembler.m
//

#import "SDLProtocolMessageAssembler.h"

#import "SDLLogMacros.h"
#import "SDLProtocolHeader.h"
#import "SDLProtocolMessage.h"

NS_ASSUME_NONNULL_BEGIN

@interface SDLProtocolMessageAssembler ()

@property (nonatomic, nullable, strong) NSMutableDictionary<NSNumber *, NSData *> *parts;

@end

@implementation SDLProtocolMessageAssembler

- (instancetype)initWithSessionID:(UInt8)sessionID {
    if (self = [super init]) {
        _sessionID = sessionID;
    }
    return self;
}

- (void)handleMessage:(SDLProtocolMessage *)message withCompletionHandler:(SDLMessageAssemblyCompletionHandler)completionHandler {
    // Validate input
    if (message.header.sessionID != self.sessionID) {
        SDLLogE(@"Message part sent to the wrong assembler. This session id: %d, message session id: %d", self.sessionID, message.header.sessionID);
        return;
    }

    if (self.parts == nil) {
        self.parts = [NSMutableDictionary dictionary];
    }


    // Determine which frame it is and save it.
    // Note: frames are numbered 1,2,3, ..., 0. Always 0 for last frame.
    if (message.header.frameType == SDLFrameTypeFirst) {
        // If it's the first-frame, extract the meta-data
        self.expectedBytes = NSSwapBigIntToHost(((UInt32 *)message.payload.bytes)[0]);
        self.frameCount = NSSwapBigIntToHost(((UInt32 *)message.payload.bytes)[1]);
        self.parts[@"firstframe"] = message.payload;
    } else if (message.header.frameType == SDLFrameTypeConsecutive) {
        // Save the frame w/ frame# as the key
        NSInteger frameNumber = message.header.frameData;
        NSNumber *frameNumberObj = @(frameNumber);
        self.parts[frameNumberObj] = message.payload;
    }

    // If we have all the parts, assemble it and execute the completion handler.
    SDLProtocolMessage *assembledMessage = nil;
    if (self.parts.count == self.frameCount + 1) { // +1 since we also require the first-frame
        // Create the header
        SDLProtocolHeader *header = message.header.copy;
        header.frameType = SDLFrameTypeSingle;
        header.frameData = SDLFrameInfoSingleFrame;

        // Create the payload
        NSMutableData *payload = [NSMutableData data];
        for (unsigned int i = 1; i < self.frameCount; i++) {
            NSData *dataToAppend = self.parts[@(i)];
            [payload appendData:dataToAppend];
        }

        // Append the last frame, it has a frame # of 0.
        NSData *dataToAppend = self.parts[@0];
        [payload appendData:dataToAppend];

        // Validation
        if (payload.length != self.expectedBytes) {
            SDLLogW(@"Collected bytes size of %lu not equal to expected size of %i.", (unsigned long)payload.length, (unsigned int)self.expectedBytes);
        }

        // Create the message.
        assembledMessage = [SDLProtocolMessage messageWithHeader:header andPayload:payload];

        // Execute completion handler.
        if (completionHandler != nil) {
            completionHandler(YES, assembledMessage);
        }

        // Done with this data, release it.
        self.parts = nil;
    } else {
        // Not done, let caller know
        if (completionHandler != nil) {
            completionHandler(NO, nil);
        }
    }
}

@end

NS_ASSUME_NONNULL_END