xref: /rockchip-linux_mpp/mpp/base/mpp_bitput.c (revision 437bfbeb9567cca9cd9080e3f6954aa9d6a94f18)
1*437bfbebSnyanmisaka /*
2*437bfbebSnyanmisaka  *
3*437bfbebSnyanmisaka  * Copyright 2015 Rockchip Electronics Co. LTD
4*437bfbebSnyanmisaka  *
5*437bfbebSnyanmisaka  * Licensed under the Apache License, Version 2.0 (the "License");
6*437bfbebSnyanmisaka  * you may not use this file except in compliance with the License.
7*437bfbebSnyanmisaka  * You may obtain a copy of the License at
8*437bfbebSnyanmisaka  *
9*437bfbebSnyanmisaka  *      http://www.apache.org/licenses/LICENSE-2.0
10*437bfbebSnyanmisaka  *
11*437bfbebSnyanmisaka  * Unless required by applicable law or agreed to in writing, software
12*437bfbebSnyanmisaka  * distributed under the License is distributed on an "AS IS" BASIS,
13*437bfbebSnyanmisaka  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*437bfbebSnyanmisaka  * See the License for the specific language governing permissions and
15*437bfbebSnyanmisaka  * limitations under the License.
16*437bfbebSnyanmisaka  */
17*437bfbebSnyanmisaka 
18*437bfbebSnyanmisaka #include <string.h>
19*437bfbebSnyanmisaka #include "mpp_bitput.h"
20*437bfbebSnyanmisaka 
mpp_set_bitput_ctx(BitputCtx_t * bp,RK_U64 * data,RK_U32 len)21*437bfbebSnyanmisaka RK_S32 mpp_set_bitput_ctx(BitputCtx_t *bp, RK_U64 *data, RK_U32 len)
22*437bfbebSnyanmisaka {
23*437bfbebSnyanmisaka     memset(bp, 0, sizeof(BitputCtx_t));
24*437bfbebSnyanmisaka     bp->index  = 0;
25*437bfbebSnyanmisaka     bp->bitpos = 0;
26*437bfbebSnyanmisaka     bp->bvalue = 0;
27*437bfbebSnyanmisaka     bp->size   = len;
28*437bfbebSnyanmisaka     bp->buflen = len;  // align 64bit
29*437bfbebSnyanmisaka     bp->pbuf   = data;
30*437bfbebSnyanmisaka     return 0;
31*437bfbebSnyanmisaka }
32*437bfbebSnyanmisaka 
mpp_put_bits(BitputCtx_t * bp,RK_U64 invalue,RK_S32 lbits)33*437bfbebSnyanmisaka void mpp_put_bits(BitputCtx_t *bp, RK_U64 invalue, RK_S32 lbits)
34*437bfbebSnyanmisaka {
35*437bfbebSnyanmisaka     RK_U8 hbits = 0;
36*437bfbebSnyanmisaka 
37*437bfbebSnyanmisaka     if (!lbits) return;
38*437bfbebSnyanmisaka 
39*437bfbebSnyanmisaka     if (bp->index >= bp->buflen) return;
40*437bfbebSnyanmisaka 
41*437bfbebSnyanmisaka     hbits = 64 - lbits;
42*437bfbebSnyanmisaka     invalue = (invalue << hbits) >> hbits;
43*437bfbebSnyanmisaka     bp->bvalue |= invalue << bp->bitpos;  // high bits value
44*437bfbebSnyanmisaka     if ((bp->bitpos + lbits) >= 64) {
45*437bfbebSnyanmisaka         bp->pbuf[bp->index] = bp->bvalue;
46*437bfbebSnyanmisaka         bp->bvalue = invalue >> (64 - bp->bitpos);  // low bits value
47*437bfbebSnyanmisaka         bp->index++;
48*437bfbebSnyanmisaka     }
49*437bfbebSnyanmisaka 
50*437bfbebSnyanmisaka     if (bp->index >= bp->buflen)
51*437bfbebSnyanmisaka         return;
52*437bfbebSnyanmisaka 
53*437bfbebSnyanmisaka     bp->pbuf[bp->index] = bp->bvalue;
54*437bfbebSnyanmisaka     bp->bitpos = (bp->bitpos + lbits) & 63;
55*437bfbebSnyanmisaka     // mpp_log("bp->index = %d bp->bitpos = %d lbits = %d invalue 0x%x bp->hvalue 0x%x  bp->lvalue 0x%x",bp->index,bp->bitpos,lbits, (RK_U32)invalue,(RK_U32)(bp->bvalue >> 32),(RK_U32)bp->bvalue);
56*437bfbebSnyanmisaka }
57*437bfbebSnyanmisaka 
mpp_put_align(BitputCtx_t * bp,RK_S32 align_bits,int flag)58*437bfbebSnyanmisaka void mpp_put_align(BitputCtx_t *bp, RK_S32 align_bits, int flag)
59*437bfbebSnyanmisaka {
60*437bfbebSnyanmisaka     RK_U32 word_offset = 0,  len = 0;
61*437bfbebSnyanmisaka 
62*437bfbebSnyanmisaka     word_offset = (align_bits >= 64) ? ((bp->index & (((align_bits & 0xfe0) >> 6) - 1)) << 6) : 0;
63*437bfbebSnyanmisaka     len = (align_bits - (word_offset + (bp->bitpos % align_bits))) % align_bits;
64*437bfbebSnyanmisaka     while (len > 0) {
65*437bfbebSnyanmisaka         if (len >= 8) {
66*437bfbebSnyanmisaka             if (flag == 0)
67*437bfbebSnyanmisaka                 mpp_put_bits(bp, ((RK_U64)0 << (64 - 8)) >> (64 - 8), 8);
68*437bfbebSnyanmisaka             else
69*437bfbebSnyanmisaka                 mpp_put_bits(bp, (0xffffffffffffffff << (64 - 8)) >> (64 - 8), 8);
70*437bfbebSnyanmisaka             len -= 8;
71*437bfbebSnyanmisaka         } else {
72*437bfbebSnyanmisaka             if (flag == 0)
73*437bfbebSnyanmisaka                 mpp_put_bits(bp, ((RK_U64)0 << (64 - len)) >> (64 - len), len);
74*437bfbebSnyanmisaka             else
75*437bfbebSnyanmisaka                 mpp_put_bits(bp, (0xffffffffffffffff << (64 - len)) >> (64 - len), len);
76*437bfbebSnyanmisaka             len -= len;
77*437bfbebSnyanmisaka         }
78*437bfbebSnyanmisaka     }
79*437bfbebSnyanmisaka }
80*437bfbebSnyanmisaka 
81