1 #!/usr/bin/python 2 3 # Copyright 2014 The Chromium OS Authors. All rights reserved. 4 # Use of this source code is governed by a BSD-style license that can be 5 # found in the LICENSE file. 6 7 # Reference[1]: IT680x example code: 8 # https://drive.google.com/corp/drive/u/0/folders/0B8Lcp5hqbjaqaE5WdDA5alVWOXc 9 10 # Reference[2]: IT6803 Programming Guide: 11 # https://docs.google.com/viewer?a=v&pid=sites&srcid=\ 12 # Y2hyb21pdW0ub3JnfGRldnxneDoyNGVmNGFiMDE4ZWJiZDM2 13 14 # This code is a library for using IT680X chip in chameleon. 15 16 import sys 17 import util 18 from time import sleep 19 20 usage = """\ 21 Usage: 22 it6803 -- print command usage 23 it6803 cec_reg_print -- print all cec registers value 24 it6803 cec_msg_receive -- print receiving cec message 25 it6803 cec_msg {cmd} -- send cec message 26 """ 27 28 QUEUE_SIZE = 3 29 q_head = 0 30 q_tail = 0 31 regTxOutState = 3 32 33 logicalAddr = 0 34 initiatorAddr = 0x0F 35 cecTxState = 0 36 37 txCmdBuf = [0x00] * 19 38 rxCecBuf = [0x00] * 19 39 queue = [[0x00 for i in range(19)] for j in range(QUEUE_SIZE)] 40 41 # Chameleon register address 42 I2C_HDMI = 0x48 43 I2C_CEC = 0x4A 44 45 # Chameleon CEC control registers 46 # (name starts with REG is register addr, followings are values for this reg) 47 REG06 = 0x06 48 REG_EMPTY = 0x00 49 REG07 = 0x07 50 ENABLE_CEC_INTERRUPT_PIN = 0x40 51 52 REG08 = 0x08 53 FIRE_FRAME = 0x80 54 DEBUG_CEC_CLEAR = 0x40 55 CEC_SCHMITT_TRIGGER = 0x08 56 CEC_INTERRUPT = 0x01 57 58 REG09 = 0x09 59 REGION_SELECT = 0x40 60 INITAITOR_RX_CEC = 0x20 61 ACKNOWLEDGE = 0x01 62 63 REG_MIN_BIT = 0x0B 64 REG_TIME_UNIT = 0x0C 65 66 REG0F = 0x0F 67 IO_PULL_UP = 0x50 68 69 REG_TARG_ADDR = 0x22 70 REG_MSCOUNT_L = 0x45 71 REG_MSCOUNT_M = 0x46 72 REG_MSCOUNT_H = 0x47 73 REF_INT_STATUS= 0x4C 74 75 def main(cmdline): 76 """ Main function. """ 77 args = [''] * 4 78 for i, x in enumerate(cmdline): 79 args[i] = x 80 cmd = args[1] 81 82 if cmd == '': cmd = 'help' 83 fname = 'cmd_' + cmd 84 85 cec_open() 86 if fname in globals(): 87 if args[2] == '': 88 globals()[fname]() 89 else: 90 globals()[fname](args[2]) 91 else: 92 print 'Unknown command', cmd 93 cec_close() 94 95 96 def cmd_help(): 97 """ Print help message. """ 98 print usage 99 100 101 def cec_open(): 102 """ Enable cec port. """ 103 # enable IT6803 CEC port: enable cec clock and assign slave addr 104 i2cset(I2C_HDMI, 0x0E, 0xFF) 105 i2cset(I2C_HDMI, 0x86, 0x95) 106 107 def cec_close(): 108 """ Close cec port. """ 109 # disable cec slave addr 110 i2cset(I2C_HDMI, 0x86, 0x94) 111 112 113 def cec_init(): 114 """ Initialize cec port in chameleon. """ 115 # initial CEC register. From reference[1] Ln480 116 117 # enable it680x cec 118 i2cset(I2C_CEC, 0xF8, 0xC3) 119 i2cset(I2C_CEC, 0xF8, 0xA5) 120 q_head = 0 121 q_tail = 0 122 regTxOutState = 3 123 124 # get 100ms timer, according to ref [1,2] 125 i2cset(I2C_CEC, REG09, ACKNOWLEDGE) 126 sleep(0.099) 127 i2cset(I2C_CEC, REG09, REG_EMPTY) 128 high = util.i2c_read(0, I2C_CEC, REG_MSCOUNT_H, 1)[0] * 0x10000 129 mid = util.i2c_read(0, I2C_CEC, REG_MSCOUNT_M, 1)[0] * 0x100 130 low = util.i2c_read(0, I2C_CEC, REG_MSCOUNT_L, 1)[0] 131 tus = (high + mid + low) / 1000 132 # print tus 133 134 # CEC configuration 135 i2cset(I2C_CEC, REG09, INITAITOR_RX_CEC | REGION_SELECT) 136 i2cset(I2C_CEC, REG_MIN_BIT, 0x14) 137 i2cset(I2C_CEC, REG_TIME_UNIT, tus) 138 i2cset(I2C_CEC, REG_TARG_ADDR, logicalAddr) 139 i2cset(I2C_CEC, REG08, CEC_SCHMITT_TRIGGER) 140 uc = util.i2c_read(0, I2C_CEC, REG09, 1)[0] 141 # i2cset(I2C_CEC, REG09, uc|0x02) 142 # cec_clr_int 143 i2cset(I2C_CEC, REG08, CEC_INTERRUPT|DEBUG_CEC_CLEAR|CEC_SCHMITT_TRIGGER) 144 i2cset(I2C_CEC, REG08, CEC_SCHMITT_TRIGGER|DEBUG_CEC_CLEAR) 145 # print 'logicalAddr: {}, TimeUnit: {}'.format(logicalAddr,tus) 146 147 # Enable CEC interrupt pin 148 reg07_val = util.i2c_read(0, I2C_CEC, REG07, 1)[0] 149 i2cset(I2C_CEC, REG07, reg07_val | ENABLE_CEC_INTERRUPT_PIN) 150 151 # Enable ALL interrupt mask 152 i2cset(I2C_CEC, REG06, REG_EMPTY) 153 154 # IO pull up enable 155 i2cset(I2C_CEC, REG0F, IO_PULL_UP) 156 157 def cec_msg_receive(): 158 """ Read message received. """ 159 # 0x3F means all interrupts are on 160 cecInt = cec_reg_read(REF_INT_STATUS) & 0x3F 161 if 0 != (cecInt & 0x10): 162 if not cec_msg_read(): 163 raise Exception('Queue is full!') 164 ## TODO check interrupt register Status 165 i2c_cec_set(REF_INT_STATUS, cecInt) 166 # Decode received message 167 return cec_decode() 168 169 170 def cmd_cec_msg(message): 171 """ parent function for a cec message. """ 172 cec_init() 173 fname = 'cec_msg_' + message 174 globals()[fname]() 175 cec_transmit() 176 177 def cec_msg_standby(): 178 """ Send a stand by message. """ 179 # F = boardcast, 0x36 = stand by message 180 cec_cmd_set(0xF, 0x36, None, None) 181 # other operations need more assignments 182 183 def cec_msg_viewon(): 184 """ Send a view on message. """ 185 # 0 = TV, 0x04 = image on 186 cec_cmd_set(0x0, 0x04, None, None) 187 188 def cec_msg_poweron(): 189 """ Make a power on cec message. """ 190 global initiatorAddr 191 # 0x90 = power status message 192 cec_cmd_set(initiatorAddr, 0x90, 0x00, None) 193 194 def cec_msg_poweroff(): 195 """ Make a power off cec message. """ 196 global initiatorAddr 197 # 0x90 = power status message 198 cec_cmd_set(initiatorAddr, 0x90, 0x01, None) 199 200 def cec_reg_read(offset): 201 """ read it6803's register value from i2c line. """ 202 return util.i2c_read(0, I2C_CEC, offset, 1)[0] 203 204 def cec_cmd_set(follower, txCmd, operand1, operand2): 205 """ Compose a cec message. """ 206 # print 'follower: {}, cmd: {}'.format(follower, txCmd) 207 # TODO set variables 208 txCmdBuf[0] = 2 209 txCmdBuf[1] = (logicalAddr<<4) + follower 210 txCmdBuf[2] = txCmd 211 txCmdBuf[3] = 0 212 txCmdBuf[4] = 0 213 if operand1 is not None: 214 txCmdBuf[3] = operand1 215 txCmdBuf[0] = 3 216 if operand2 is not None: 217 txCmdBuf[4] = operand2 218 txCmdBuf[0] = 4 219 # print txCmdBuf 220 return 221 222 def cec_transmit(): 223 """ File a cec message out. """ 224 # Assume the state is cecTransfer 225 # Set values from 0x10 to 0x23 226 i2c_cec_set(0x23, txCmdBuf[0]) 227 for i in range (0, txCmdBuf[0]): 228 i2c_cec_set(0x10+i, txCmdBuf[i+1]) 229 230 # Fire command 231 i2c_cec_set(REG08, FIRE_FRAME | CEC_SCHMITT_TRIGGER | DEBUG_CEC_CLEAR) 232 i2c_cec_set(REG08, CEC_SCHMITT_TRIGGER | DEBUG_CEC_CLEAR) 233 return 234 235 def cec_msg_read(): 236 """ Read incoming cec messages from memory. """ 237 global q_head, q_tail 238 if (q_head % QUEUE_SIZE) != (q_tail % QUEUE_SIZE): 239 return False 240 q_tail += 1 241 i = q_tail % QUEUE_SIZE 242 # 0x30 is starting point for receiving message 243 data = util.i2c_read(0, I2C_CEC, 0x30, 19) 244 for j in range(1, 19): 245 queue[i][j] = data[j-1] 246 queue[i][0] = data[18] 247 return True 248 249 def cec_decode(): 250 """ Process incoming cec message. """ 251 global q_head, q_tail, initiatorAddr 252 if (q_head % QUEUE_SIZE) == (q_tail % QUEUE_SIZE): 253 # Queue is empty 254 return 255 q_head += 1 256 rxCecBuf = queue[q_head % QUEUE_SIZE] 257 #print rxCecBuf 258 259 if (rxCecBuf[0] == 1): 260 if logicalAddr == (rxCecBuf[1] & 0x0F): 261 # eReportPhysicalAddress 262 return 263 # Validate message 264 initiatorAddr = (rxCecBuf[1] >> 4) & 0x0F 265 followerAddr = rxCecBuf[1] & 0x0F 266 print 'Initiator: {} Follower: {}'.format(initiatorAddr, followerAddr) 267 268 if (rxCecBuf[2] == 0x04): 269 print 'received image-view-on' 270 elif (rxCecBuf[2] == 0x36): 271 print 'received standby' 272 else: 273 print 'other command: {}'.format(rxCecBuf[2]) 274 return rxCecBuf[2] 275 276 def i2cset(addr, offset, value): 277 """ set some register value via i2c line. """ 278 util.i2c_write(0, addr, offset, [value]) 279 280 def i2c_cec_set(offset, value): 281 """ set it6803's register value via i2c line. """ 282 i2cset(I2C_CEC, offset, value) 283 284 if __name__ == '__main__': 285 main(sys.argv) 286