-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhuffEnc.m
81 lines (75 loc) · 2.91 KB
/
huffEnc.m
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
function huffStream = huffEnc(runSymbols)
%huffEnc
%Inputs:
%runSymbols: Matrix contain pairs of (precedingZeros, quantSymbol). [R-by-2]
%return:
%huffStream: A matrix of uint8 (bytes) contains the huffman code.
%
% Each step is explained later.
%
if ~ismatrix(runSymbols), error('Error. The argument {runSymbols} must be a Rx2 matrix.'); end
global DC_Huff_L DC_Huff_C AC_Huff_L AC_Huff_C isLuminance;
if isLuminance
DC_Huff = DC_Huff_L;
AC_Huff = AC_Huff_L;
else
DC_Huff = DC_Huff_C;
AC_Huff = AC_Huff_C;
end
[RowNumber, ~] = size(runSymbols); strHuff = [];
%~ First, match the symbol with the 'Category' ~%
% The symbol belongs to 'Category' when the following apply:
% -2^(Category) < Symbol < 2^(Category)
%~ Category and Additional Bits for DC coefficient ~%
Category = 0; % Category = 0, ONLY FOR Symbol = 0
if runSymbols(1,2)
Category = floor( log2( abs(runSymbols(1,2)) ) ) + 1; % Table F.1
end
DC_Magn = Category;
dcAdditionalBits = getBinary(runSymbols(1,2), Category);
%~ Category and Additional Bits for AC coefficients ~%
AC_Magn = zeros(RowNumber-1, 1); acAdditionalBits = string( zeros(RowNumber-1, 1));
for j = 2:RowNumber
% RunLengths [15 0] and [0 0] DOESNT HAVE any additional bits
if isequal(runSymbols(j,:), [15 0]) || isequal(runSymbols(j,:), [0 0])
AC_Magn(j-1) = 0;
acAdditionalBits(j-1) = '';
else
Category = floor( log2( abs(runSymbols(j,2)) ) ) + 1; % Table F.2
AC_Magn(j-1) = Category;
acAdditionalBits(j-1) = getBinary(runSymbols(j,2), Category);
end
end
%~ Compute Huffman code of DC Coefficient ~%
index = DC_Magn + 1;
strHuff = strcat(char(strHuff), char(DC_Huff(index)), char(dcAdditionalBits));
%~ Compute Huffman code of AC Coefficients ~%
for i = 2:RowNumber
index = runSymbols(i,1) * 10 + AC_Magn(i-1) + 1; % + 1 for the EOB
if runSymbols(i,1) == 15
index = index + 1; % + 1 for the ZRL
end
strHuff = strcat(char(strHuff), char(AC_Huff(index)), char(acAdditionalBits(i-1)));
end
%~ Transform huffman from char to bytestream (uint8) ~%
remBits = mod(length(strHuff), 8);
if remBits
addBits = string(ones(1,8-remBits));
strHuff = strcat(char(strHuff), char(strjoin(addBits(:), "")));
end
huffStream = uint8(zeros(1,length(strHuff)/8)); idx = 1;
for i = 1:8:length(strHuff)
huffStream(idx) = bin2dec(strHuff(i:i+7));
idx = idx + 1;
end
end
function binValue = getBinary(num, Category)
%~ Based on Category, produce the binary / huffman value of num ~%
if ~Category, binValue = 0; return; end % IF category = 0, then binary value is zero too.
binValue = bitget(abs(num),Category:-1:1);
if num < 0
binValue = ~binValue; % For negative values, compute the 1's complement.
end
binValue = string(binValue * 1); % Logical to int, then to string.
binValue = strjoin(binValue(:),""); % Concat the string.
end