18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * ImgTec IR Decoder setup for Sanyo protocol. 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright 2012-2014 Imagination Technologies Ltd. 68c2ecf20Sopenharmony_ci * 78c2ecf20Sopenharmony_ci * From ir-sanyo-decoder.c: 88c2ecf20Sopenharmony_ci * 98c2ecf20Sopenharmony_ci * This protocol uses the NEC protocol timings. However, data is formatted as: 108c2ecf20Sopenharmony_ci * 13 bits Custom Code 118c2ecf20Sopenharmony_ci * 13 bits NOT(Custom Code) 128c2ecf20Sopenharmony_ci * 8 bits Key data 138c2ecf20Sopenharmony_ci * 8 bits NOT(Key data) 148c2ecf20Sopenharmony_ci * 158c2ecf20Sopenharmony_ci * According with LIRC, this protocol is used on Sanyo, Aiwa and Chinon 168c2ecf20Sopenharmony_ci * Information for this protocol is available at the Sanyo LC7461 datasheet. 178c2ecf20Sopenharmony_ci */ 188c2ecf20Sopenharmony_ci 198c2ecf20Sopenharmony_ci#include "img-ir-hw.h" 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ci/* Convert Sanyo data to a scancode */ 228c2ecf20Sopenharmony_cistatic int img_ir_sanyo_scancode(int len, u64 raw, u64 enabled_protocols, 238c2ecf20Sopenharmony_ci struct img_ir_scancode_req *request) 248c2ecf20Sopenharmony_ci{ 258c2ecf20Sopenharmony_ci unsigned int addr, addr_inv, data, data_inv; 268c2ecf20Sopenharmony_ci /* a repeat code has no data */ 278c2ecf20Sopenharmony_ci if (!len) 288c2ecf20Sopenharmony_ci return IMG_IR_REPEATCODE; 298c2ecf20Sopenharmony_ci if (len != 42) 308c2ecf20Sopenharmony_ci return -EINVAL; 318c2ecf20Sopenharmony_ci addr = (raw >> 0) & 0x1fff; 328c2ecf20Sopenharmony_ci addr_inv = (raw >> 13) & 0x1fff; 338c2ecf20Sopenharmony_ci data = (raw >> 26) & 0xff; 348c2ecf20Sopenharmony_ci data_inv = (raw >> 34) & 0xff; 358c2ecf20Sopenharmony_ci /* Validate data */ 368c2ecf20Sopenharmony_ci if ((data_inv ^ data) != 0xff) 378c2ecf20Sopenharmony_ci return -EINVAL; 388c2ecf20Sopenharmony_ci /* Validate address */ 398c2ecf20Sopenharmony_ci if ((addr_inv ^ addr) != 0x1fff) 408c2ecf20Sopenharmony_ci return -EINVAL; 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_ci /* Normal Sanyo */ 438c2ecf20Sopenharmony_ci request->protocol = RC_PROTO_SANYO; 448c2ecf20Sopenharmony_ci request->scancode = addr << 8 | data; 458c2ecf20Sopenharmony_ci return IMG_IR_SCANCODE; 468c2ecf20Sopenharmony_ci} 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_ci/* Convert Sanyo scancode to Sanyo data filter */ 498c2ecf20Sopenharmony_cistatic int img_ir_sanyo_filter(const struct rc_scancode_filter *in, 508c2ecf20Sopenharmony_ci struct img_ir_filter *out, u64 protocols) 518c2ecf20Sopenharmony_ci{ 528c2ecf20Sopenharmony_ci unsigned int addr, addr_inv, data, data_inv; 538c2ecf20Sopenharmony_ci unsigned int addr_m, data_m; 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ci data = in->data & 0xff; 568c2ecf20Sopenharmony_ci data_m = in->mask & 0xff; 578c2ecf20Sopenharmony_ci data_inv = data ^ 0xff; 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci if (in->data & 0xff700000) 608c2ecf20Sopenharmony_ci return -EINVAL; 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_ci addr = (in->data >> 8) & 0x1fff; 638c2ecf20Sopenharmony_ci addr_m = (in->mask >> 8) & 0x1fff; 648c2ecf20Sopenharmony_ci addr_inv = addr ^ 0x1fff; 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci out->data = (u64)data_inv << 34 | 678c2ecf20Sopenharmony_ci (u64)data << 26 | 688c2ecf20Sopenharmony_ci addr_inv << 13 | 698c2ecf20Sopenharmony_ci addr; 708c2ecf20Sopenharmony_ci out->mask = (u64)data_m << 34 | 718c2ecf20Sopenharmony_ci (u64)data_m << 26 | 728c2ecf20Sopenharmony_ci addr_m << 13 | 738c2ecf20Sopenharmony_ci addr_m; 748c2ecf20Sopenharmony_ci return 0; 758c2ecf20Sopenharmony_ci} 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci/* Sanyo decoder */ 788c2ecf20Sopenharmony_cistruct img_ir_decoder img_ir_sanyo = { 798c2ecf20Sopenharmony_ci .type = RC_PROTO_BIT_SANYO, 808c2ecf20Sopenharmony_ci .control = { 818c2ecf20Sopenharmony_ci .decoden = 1, 828c2ecf20Sopenharmony_ci .code_type = IMG_IR_CODETYPE_PULSEDIST, 838c2ecf20Sopenharmony_ci }, 848c2ecf20Sopenharmony_ci /* main timings */ 858c2ecf20Sopenharmony_ci .unit = 562500, /* 562.5 us */ 868c2ecf20Sopenharmony_ci .timings = { 878c2ecf20Sopenharmony_ci /* leader symbol */ 888c2ecf20Sopenharmony_ci .ldr = { 898c2ecf20Sopenharmony_ci .pulse = { 16 /* 9ms */ }, 908c2ecf20Sopenharmony_ci .space = { 8 /* 4.5ms */ }, 918c2ecf20Sopenharmony_ci }, 928c2ecf20Sopenharmony_ci /* 0 symbol */ 938c2ecf20Sopenharmony_ci .s00 = { 948c2ecf20Sopenharmony_ci .pulse = { 1 /* 562.5 us */ }, 958c2ecf20Sopenharmony_ci .space = { 1 /* 562.5 us */ }, 968c2ecf20Sopenharmony_ci }, 978c2ecf20Sopenharmony_ci /* 1 symbol */ 988c2ecf20Sopenharmony_ci .s01 = { 998c2ecf20Sopenharmony_ci .pulse = { 1 /* 562.5 us */ }, 1008c2ecf20Sopenharmony_ci .space = { 3 /* 1687.5 us */ }, 1018c2ecf20Sopenharmony_ci }, 1028c2ecf20Sopenharmony_ci /* free time */ 1038c2ecf20Sopenharmony_ci .ft = { 1048c2ecf20Sopenharmony_ci .minlen = 42, 1058c2ecf20Sopenharmony_ci .maxlen = 42, 1068c2ecf20Sopenharmony_ci .ft_min = 10, /* 5.625 ms */ 1078c2ecf20Sopenharmony_ci }, 1088c2ecf20Sopenharmony_ci }, 1098c2ecf20Sopenharmony_ci /* repeat codes */ 1108c2ecf20Sopenharmony_ci .repeat = 108, /* 108 ms */ 1118c2ecf20Sopenharmony_ci .rtimings = { 1128c2ecf20Sopenharmony_ci /* leader symbol */ 1138c2ecf20Sopenharmony_ci .ldr = { 1148c2ecf20Sopenharmony_ci .space = { 4 /* 2.25 ms */ }, 1158c2ecf20Sopenharmony_ci }, 1168c2ecf20Sopenharmony_ci /* free time */ 1178c2ecf20Sopenharmony_ci .ft = { 1188c2ecf20Sopenharmony_ci .minlen = 0, /* repeat code has no data */ 1198c2ecf20Sopenharmony_ci .maxlen = 0, 1208c2ecf20Sopenharmony_ci }, 1218c2ecf20Sopenharmony_ci }, 1228c2ecf20Sopenharmony_ci /* scancode logic */ 1238c2ecf20Sopenharmony_ci .scancode = img_ir_sanyo_scancode, 1248c2ecf20Sopenharmony_ci .filter = img_ir_sanyo_filter, 1258c2ecf20Sopenharmony_ci}; 126