From patchwork Tue Apr 2 10:49:27 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Allen_Lin X-Patchwork-Id: 785572 Received: from APC01-TYZ-obe.outbound.protection.outlook.com (mail-tyzapc01olkn2074.outbound.protection.outlook.com [40.92.107.74]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id F0446605B6; Tue, 2 Apr 2024 10:50:07 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=fail smtp.client-ip=40.92.107.74 ARC-Seal: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055009; cv=fail; b=RrBBhbNk4bn9mDo09R8kOwZo/ZILFnVYd8uzlIwHSDg7pW8N+fUKC1zNVOjLXy9E+fvAuWRJUzZA6KnlkKN+dPabnc6Of2oiaCBj2IMTamxGwuDfQwcRscxN4ZbcPKAdRbSgyQ1YAg/XeVUpGiWPMJYm4XltO81TC+Rbt9m+f+0= ARC-Message-Signature: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055009; c=relaxed/simple; bh=jCTMoJQjnbuB2X3DycG5cnYK3Hme4YNYsdrxf3XDJ/g=; h=From:To:Cc:Subject:Date:Message-ID:In-Reply-To:References: Content-Type:MIME-Version; b=PYyBDDzKVSHj2zvPEv+6y/fpgaE1Cgv3j7rDlXhmFYjkz0sbClcQBLSaIVx8x/Xsyc76+JKmIkfH9eCUGtH6BIrmBBEyiYsdzdBDdnchPbyZkzWrROcSyIeopkzeMmG8rOyC2v+NEDHGLBdDNOJ/EJA9nBwY0u3rFxu79PFms1Q= ARC-Authentication-Results: i=2; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com; spf=pass smtp.mailfrom=hotmail.com; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b=umlzAGxE; arc=fail smtp.client-ip=40.92.107.74 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=hotmail.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b="umlzAGxE" ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=LG5WZdNZ2LWimLHDmnECnt5zzJm9ALdzocomTcrgPzql17CAgoekKipyFM6zEH4CoRpgsDdERkL3OdPkM0kHt0Z6VCPVUDPJXRLBd5jKoq1Oymym7xsbuAR+Oq2BmwrMGaeWlKIclIIFcSe5PXntsFQaJ96oBHJaMQlGo5bxJvo9PmmecI7h5ZRndWDpedkrx96aVbL8DQtdrKsWo79AxFWlahHQ672u15ubsBFAg/jR9O5TipvkRC1bOkhVa/u+dMW2pYphjsUhxF0bbUK5Q8KbEDV5RzbTEcQ4cJQkOb9kX9tZeQDUKhZbMe/Tgi3vULQG9uqy4zjuLOLvny/TKw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=ZAJ/Bo+VRExAfvHYPdxG54EZGxUNoZ5ZWBsdzxxhroQ=; b=beTOPBjFDttjXDeY/wqrcokqnvFnohhrBa87ROJw9FteI+Tl1gY2L2AooRQnSihiiTGKEivDa5KtLymBU4mbU1k/Nv4ei0W6r1g5Ee0Lh6BXhkyL/gQJUnK0fyUa90JCrpBhL5x9QjXs6b0DVqLGlwqu0CMHvOo5nqiYk6L+KTvIo4HhB9aDVKfHCbbW54ehSOtHIsiZX18Gc52agS8/BgGGzO0fiGr/a8kDXfYpAodC8C5QlPR2NIM6LaDfrDPJp0plUX4H+Bt3YNskaMEFNRieK+Q7xssuXBIrJAFnA/YZBbj9V7/S6JV3BWleLljw5NSiRJsN6qBZIjbHmVlsrA== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=none; dmarc=none; dkim=none; arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=hotmail.com; s=selector1; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=ZAJ/Bo+VRExAfvHYPdxG54EZGxUNoZ5ZWBsdzxxhroQ=; b=umlzAGxEpIhFOoQ5MIvqEAVf+NqXYQr6jxcIK09GQXfxIB0abyfM18AYKZWuK1PldMPelzW/3jZZ5uahlk8zivAr3ugSffObaYdEdUj1PP1pR4DeMYQoyjwH1oqqC9heZ7cuaUHNIa5192D0UnrTC5ZkMOubO/dA5hu+YG9pF2myng3rLzloxpBoJuCXlu7SuSVeq8cT+tdYHbRBcukECSbWHYfw4MZJplB9mnJx0PYLT4KKwFyFdS6aeyJ33NWS7MWE7MlD43rX2iibVcKSXRxrbei+F2qEIGdXK9M9Cs5lKqVuU8F83Fj4jNKt858lxLdE7XsGAsAbAe7i0HnR6Q== Received: from TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) by KL1PR06MB6964.apcprd06.prod.outlook.com (2603:1096:820:121::11) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.7409.46; Tue, 2 Apr 2024 10:50:04 +0000 Received: from TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf]) by TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf%5]) with mapi id 15.20.7409.042; Tue, 2 Apr 2024 10:50:04 +0000 From: Allen_Lin To: dmitry.torokhov@gmail.com, robh@kernel.org, krzysztof.kozlowski+dt@linaro.org, conor@kernel.org, jikos@kernel.org, benjamin.tissoires@redhat.com, linux-input@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Cc: Allen_Lin Subject: [PATCH v2 1/4] dt-bindings: input: Add Himax HX83102J touchscreen Date: Tue, 2 Apr 2024 18:49:27 +0800 Message-ID: X-Mailer: git-send-email 2.34.1 In-Reply-To: <20240402104930.1053016-1-allencl_lin@hotmail.com> References: <20240402104930.1053016-1-allencl_lin@hotmail.com> X-TMN: [Qxv13yMcFL/mP+t9E2SGz3lG5huSO6Vf] X-ClientProxiedBy: PS2PR02CA0031.apcprd02.prod.outlook.com (2603:1096:300:59::19) To TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) X-Microsoft-Original-Message-ID: <20240402104930.1053016-2-allencl_lin@hotmail.com> Precedence: bulk X-Mailing-List: linux-input@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 X-MS-Exchange-MessageSentRepresentingType: 1 X-MS-PublicTrafficType: Email X-MS-TrafficTypeDiagnostic: TY0PR06MB5611:EE_|KL1PR06MB6964:EE_ X-MS-Office365-Filtering-Correlation-Id: f83b2aff-e932-4cdf-479a-08dc5302a736 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: Tbk36P0Jsgmf0CwojoyiRpEyUD0SyA/zDbsmkOHNd6lozfgikdXX/nz/g6zQaFx9mAU8C9hr+k2xVjyzWbsXz3gOnYD6YLV9wksSZUkWJnjaUTSQ4+S2yvnokDSKGRPbhbrGdieCjkB0y3iM9yx5D9FjIP7FaKmPM/7jZjV0y1pfynNR7rgEt0e1Qg9luTN2jPRMjlG8zB8kiQEJm2Rt31k/jYqDKiF0NUxpB8o538oJxk2Y15ICVbNAptdPc3btOMAmnI4Hi0uKo/NleA3pRQR/O5GrFhDV9jFmjRXTVEBYA3feASxwB0m6c7h5AeyUcFCG4HxbPHueIYrug+bxHh+SL4vMfRrBHaTAtOCUF0stbbjQ0Ak2ihm7IvcnXih60sXtWojjLut+9+ICYgFEpI8n48yn7JLRq27Dihk6ZeYQ7e2qkjn2lTQ7MzYaaOpK4zQZQauwv4mHZ8LCvQkOxzW/yY5pNtPM8w6mNLadMMYZrxjLMYbj/MsTEDZ7R4EmhoGXqpm0fDGhoBr1kLrJ4vKSS445cbbRMJjhoL85EtQgVyqEpVdTbAUnkDjyQs1MDM6ZsAn31WD08aA6St6e1li+0vvzX0Rl59yv9zW7lkVpo65iboseeNIbbtV4TqOWQjdVqq1sat8rptMVq15rna7V88vH/M0MVYpFuQeLdL7Zqo3lkhvPB6JVkfkG31Xz X-MS-Exchange-AntiSpam-MessageData-ChunkCount: 1 X-MS-Exchange-AntiSpam-MessageData-0: qQJoUNKIuIPsafIZJF1eUTfqXASowBAoPxftF/F5CrHMluLDDA4gKBg7PMi6vSma7WwHApdd1ZemTZWqnFipYiaZ+5cIYLtA550fpTxfL5o0ZQDhEIdRUQkxTKobYxRAkfISzNicVWFJSLPTC+Vj3kfbEA3T0Jrhh8VbZSAcHg2fuC0d8hkNoQi/4HTb5gpR7ThYd4xSqb8EH4Kpj5oUVGI8Sfs0vkC/0oeL6B69xgW/BDNyPOlfzjAtDIpBvpSgS9EZVXS+tXG2ba0n2CRkCZ/v/1B74SAtNsi253qu2Nqo9r/tqqqZ66hR13xz61CVNqEVN2tJ6Uh8wcnSRnkQEfptT6cqkVGeIictuAygqvgW+0mdUy7Nzwoi9Su6sOurG3/3yT5csolyFdqVEe1nxUQG0PpK3/WPxSLr/nIVHJCbZnujpOP/YEbXPTL3TmGmIzh5vSiYEgjI9JYZELdUYYT9I+vm0QVYFE/H3nRT2tK5k+MGT2q/1WMFN9b1Wd7UQyinnijs1fa66fJ2p0eGxzAu4MdtvvB0SN9QWr5y9qgDR9KNeidMWhDN0QqLLlUst8jdvdQqhwWpVSTt2GKU8VXUcKcpNi4xnkYOuMQQlq/aTeMd7pIO2bHD/oOnh7AXsgnjNNWDRpVc70QK9Zobr2/M9miCMMEcyMw3f661ZrY8Q1CCPZNGtIIBEoPxfsfKMg6UQokq9ZIQllxfHH152Bc1kgna7LjW4s3OAdNG+5uOIYJN+mL1tWxV9tSIxaK6iWPPhGK6LjkSTll9EoULku0q3qc9B1TdrPLd0N74vSkHtAOwuis5kmL6OYHMdzFyHPxym9nPELKzXaD3IvhSpDkHpiFCztjmz/ayfnLe27nK92F13tzgMuibkfymFNDqL5MpJ3g9X6MyzZlRdKhVe9KpKY9NO31uz+I9JNbiCyFKOOSsn1HDpPbXrIaBfcSA/qkhj/Pp/SCTVeRtvTuI3AjJ66Lq6vJ64eGHtTkl6DPzjCQbFyXkE7x9j/Q0COVWhxPRTes3FdvF3esi3xTWPM7OFZoJIWJh13jg+9woIo5RbqJNEXU5bcLuTq/tUGHn9BrTfEmNvEkinBLT2bsua33ddSuyoXmQxQvx6ddVn016/OigKZOYOkbLX2eLcr34S7DhAhyU0TEvSHCOr7YjmYYpp7gEhF7MzaJaUDO8jeWcgRTZzcRBqFTmJ7t/3tqnABZqApGchGULw9vhccwMJ6czEsokieyDyFFAcRNzu6c= X-OriginatorOrg: sct-15-20-4755-11-msonline-outlook-3208f.templateTenant X-MS-Exchange-CrossTenant-Network-Message-Id: f83b2aff-e932-4cdf-479a-08dc5302a736 X-MS-Exchange-CrossTenant-AuthSource: TY0PR06MB5611.apcprd06.prod.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Internal X-MS-Exchange-CrossTenant-OriginalArrivalTime: 02 Apr 2024 10:50:04.3251 (UTC) X-MS-Exchange-CrossTenant-FromEntityHeader: Hosted X-MS-Exchange-CrossTenant-Id: 84df9e7f-e9f6-40af-b435-aaaaaaaaaaaa X-MS-Exchange-CrossTenant-RMS-PersistedConsumerOrg: 00000000-0000-0000-0000-000000000000 X-MS-Exchange-Transport-CrossTenantHeadersStamped: KL1PR06MB6964 Add the HX83102j touchscreen device tree bindings documents. HX83102j is a Himax TDDI touchscreen controller. It's power sequence should be bound with a lcm driver, thus it needs to be a panel follower. Others are the same as normal SPI touchscreen controller. Signed-off-by: Allen_Lin --- .../input/touchscreen/himax,hx83102j.yaml | 100 ++++++++++++++++++ MAINTAINERS | 6 ++ 2 files changed, 106 insertions(+) create mode 100644 Documentation/devicetree/bindings/input/touchscreen/himax,hx83102j.yaml diff --git a/Documentation/devicetree/bindings/input/touchscreen/himax,hx83102j.yaml b/Documentation/devicetree/bindings/input/touchscreen/himax,hx83102j.yaml new file mode 100644 index 000000000000..fe79129f704a --- /dev/null +++ b/Documentation/devicetree/bindings/input/touchscreen/himax,hx83102j.yaml @@ -0,0 +1,100 @@ +# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause) +%YAML 1.2 +--- +$id: http://devicetree.org/schemas/input/touchscreen/himax,hx83102j.yaml# +$schema: http://devicetree.org/meta-schemas/core.yaml# + +title: Himax hx83102j touchscreen + +maintainers: + - Allen Lin + +description: + This Himax hx83102j touchscreen uses the spi protocol. + +allOf: + - $ref: /schemas/input/touchscreen/touchscreen.yaml# + - $ref: /schemas/spi/spi-peripheral-props.yaml# + +properties: + compatible: + const: himax,hx83102j + + reg: + maxItems: 1 + + interrupts: + maxItems: 1 + + reset-gpios: + maxItems: 1 + + vccd-supply: + description: A phandle for the regualtor supplying IO power. + + vsn-supply: + description: Negative supply regulator. + + vsp-supply: + description: Positive supply regulator. + + ddreset-gpios: + description: A phandle of gpio for display reset controlled by the LCD driver. + This is the master reset, if this reset is triggered, the TP reset will + also be triggered. + + spi-cpha: true + + spi-cpol: true + + spi-max-frequency: true + + panel: true + + himax,firmware-name: + $ref: /schemas/types.yaml#/definitions/string + description: + Specify the file name for firmware loading. + + himax,pid: + $ref: /schemas/types.yaml#/definitions/uint32 + description: + PID for HID device, used to validate firmware. + +required: + - compatible + - reg + - interrupts + - reset-gpios + - panel + - vccd-supply + - vsn-supply + - vsp-supply + - ddreset-gpios + +additionalProperties: false + +examples: + - | + #include + #include + spi { + #address-cells = <1>; + #size-cells = <0>; + ap_ts: touchscreen@0 { + compatible = "himax,hx83102j"; + reg = <0>; + pinctrl-names = "default"; + pinctrl-0 = <&touch_int0 &touch_reset>; + reset-gpios = <&gpio1 8 GPIO_ACTIVE_LOW>; + spi-cpha; + spi-cpol; + interrupt-parent = <&gpio1>; + interrupts = <7 IRQ_TYPE_LEVEL_LOW>; + panel = <&panel>; + vccd-supply = <®ulator>; + vsn-supply = <®ulator>; + vsp-supply = <®ulator>; + ddreset-gpios = <&gpio1>; + }; + }; diff --git a/MAINTAINERS b/MAINTAINERS index 43b39956694a..aa51c60fd66d 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -9669,6 +9669,12 @@ L: linux-kernel@vger.kernel.org S: Maintained F: drivers/misc/hisi_hikey_usb.c +HIMAX HID HX83102J TOUCHSCREEN +M: Allen Lin +L: linux-input@vger.kernel.org +S: Maintained +F: Documentation/devicetree/bindings/input/touchscreen/himax,hx83102j.yaml + HIMAX HX83112B TOUCHSCREEN SUPPORT M: Job Noorman L: linux-input@vger.kernel.org From patchwork Tue Apr 2 10:49:28 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Allen_Lin X-Patchwork-Id: 785233 Received: from APC01-PSA-obe.outbound.protection.outlook.com (mail-psaapc01olkn2017.outbound.protection.outlook.com [40.92.52.17]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 2720967C46; Tue, 2 Apr 2024 10:50:13 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=fail smtp.client-ip=40.92.52.17 ARC-Seal: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055018; cv=fail; b=azpFnBAPy5xpJdlbcisVP3OUAHvT4oaWhUGklzIdITkhrBBipMPR9nmnzhxJtZ88Pm9LnxywmMPlsEE7/hCUM7wqLCzUkHzLBOZXztL3JhbUWv6W4dJ0zXX3o+wQvqOyvfwUrMac+ROcpjgNz46hrOvG62pGN4kuNqt7Ak4fauY= ARC-Message-Signature: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055018; c=relaxed/simple; bh=J9UwqQdOs4C10GA3PQWe2qzVY6KaYVlsLsvyr6WQRPA=; h=From:To:Cc:Subject:Date:Message-ID:In-Reply-To:References: Content-Type:MIME-Version; b=fz8egAuA8UyaV2ptvNdtm44+OPHdcUDuXb5r54JH1Ak7PbBbo0uotDfU90Hyz8LVGouyJabdsPhEM6d5sW7HWfqyGQe3XluNYbP7bcLa3e6qQ5oOn+WpzRSNM9jR2ZcokliBc8sjQP0MO9MRE1e/T4jERZd8sSfP5UeeAgD11OE= ARC-Authentication-Results: i=2; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com; spf=pass smtp.mailfrom=hotmail.com; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b=c9rLXYZp; arc=fail smtp.client-ip=40.92.52.17 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=hotmail.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b="c9rLXYZp" ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=oEypLY3x858YKZhqsMuG0OpfGMl9UFCnwjgGyZL/CC3IHkcE3ekTBUX5kJ/5EO4usm+M0exQtfUIiGeuiyngZ2qhZGl1sr2AiTXgCrLWw5RDeSFwm4t7K6J9vDl0ldaxTMG0QNeK2pEA83IHtlX+NfQoEDRSifTqPr6FpV+2ITn8Rw+jLTIMTi15XaUBLh8d/IAk6mPEBAVzhnWgYLiq4JSe+qqLrRacAPadiRv5r2+tzjea4Fxdd6Qo2ft6r2lMuTeFkhPbk3JOmdsQIk3EPzaSfVFBLmA6WHuXOFMRQQVhvOr+RD0JrmQFrVlbRVG4iF2kcp0XBwdbV2frkjidZw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=NvYRJapfOUbmaU6jydrQm/nnKa56LBYVdLbuaFL7+UI=; b=a1SYAC4nYwo8G2QWNPSsv/vhVYuhFjOB1KExKQq+9E0mFY3XynoqEDNdCoXdjYRRkxE9yCEkHQkX81GCq4YZZj5njyhH+uMWf8XG4tXNzio88QvISJ5M7JUDuXsoEXL6opCxK75ENzzhvpgslUxOuetRKELY4EqNU8M1iTUc5rdKnJkUjWSsRNApTl88OsPO588CHVf6I68uv3pt6/QD88WAvSh73gdTIG0efR3luP8gWsi2IHJxw/wexERv+IGovU9Et31CDUoadu6+cv+gN7ZQrXhUgSLS+vjagZPmNgMOFIdBNWT3ltAx0h4b+jT9vGAnrX3xn/9AnReKxF6vDg== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=none; dmarc=none; dkim=none; arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=hotmail.com; s=selector1; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=NvYRJapfOUbmaU6jydrQm/nnKa56LBYVdLbuaFL7+UI=; b=c9rLXYZpK42w2+Cyq9fDAO+I4/9ddNf/hFMrPzOcxRg72jpvRtFmj8G5fLT86wnQ1Qltvqep9hFk2OXJcXIYDN3jQhATxpRIP+S86XeU9w0JMeCF/UQa/hy3ekzLB+HjRWlsTzTO3YhzOds0VKmof1/g/ImGlz/xYshApf7+JKTtENLuvYB5T/0AG6FZHI1UeKGcI2YFZUUGL5H5llfkOCFArH9Ho3GFCJYYI9GRW6TvDPcfq0OVsmS8GoEG4jnkmM9Xq/ULNeB5k+r8nwWCzoMVf07SmCfHzYB5LzD2zkTXPfju6RAmBpdItB+dI39W+BZySn8XOi/A7iGmUrrj/A== Received: from TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) by KL1PR06MB6964.apcprd06.prod.outlook.com (2603:1096:820:121::11) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.7409.46; Tue, 2 Apr 2024 10:50:07 +0000 Received: from TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf]) by TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf%5]) with mapi id 15.20.7409.042; Tue, 2 Apr 2024 10:50:07 +0000 From: Allen_Lin To: dmitry.torokhov@gmail.com, robh@kernel.org, krzysztof.kozlowski+dt@linaro.org, conor@kernel.org, jikos@kernel.org, benjamin.tissoires@redhat.com, linux-input@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Cc: Allen_Lin Subject: [PATCH v2 2/4] HID: Add Himax HX83102J touchscreen driver Date: Tue, 2 Apr 2024 18:49:28 +0800 Message-ID: X-Mailer: git-send-email 2.34.1 In-Reply-To: <20240402104930.1053016-1-allencl_lin@hotmail.com> References: <20240402104930.1053016-1-allencl_lin@hotmail.com> X-TMN: [27Y1Kg2kw8zO5S2V4NB896CP//35c2SD] X-ClientProxiedBy: PS2PR02CA0031.apcprd02.prod.outlook.com (2603:1096:300:59::19) To TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) X-Microsoft-Original-Message-ID: <20240402104930.1053016-3-allencl_lin@hotmail.com> Precedence: bulk X-Mailing-List: linux-input@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 X-MS-Exchange-MessageSentRepresentingType: 1 X-MS-PublicTrafficType: Email X-MS-TrafficTypeDiagnostic: TY0PR06MB5611:EE_|KL1PR06MB6964:EE_ X-MS-Office365-Filtering-Correlation-Id: 8720ca03-9047-43de-fbbc-08dc5302a933 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: xGvJWWRASnsKLJ0mchZ5oeYmjbkjEPUKNQCsR2Npx3VWz1yh+dK9jrMauGqMqFteJIcePdsAUnlfd3u4IZKCq3OhAf8eNZFPfJuAkbcZyNnlYEJFXfS8HDFyeiQ5Cp/eLYA02jU/CuyFG32oaDXXPiKb5RLLEzdnB8rWGvcCLoaLmoYA4LbfFRj4c8MT9tPF9lkrUTzFAxBg8V+ys97PUjQZaYLfQmbvL6L/3ThlmC5QAQLfC98jLrXeTZpVoaCURdmkyCHs7bxXIW+QZ8+vlE8Cjke0LLwhor4NwVAheIC8s1t7yIYm3T89VHqUduHuJ1LoM34eV/ouu7L0dkA5n5HFwmnJ4FF8YRHER2g6WLCaHCx71cM3K8l+3hpVq/6ACLGdA55VMSd4LI2SjERHWo4y50EUoVYX/aE64VS3AF0fZd3pcy7MUOpbKpKxl7/rEsHe0/BlFOs3TFkZrrU3RaqQl+um935q+bjPtuzx3M0mCriGPirp0dQE5p2MWQ9euMJ2lKkVJXpfaIuqajiadnR6Rm2w6H86UHtQX3/kBioyuwNgHpPnLTc0QBQJT+PLgU3h830fqtkRfnFi9YAohRInpd+kcYLriRoAw4ih6HF1Lr5e1I5W5yNsdWs50nfX X-MS-Exchange-AntiSpam-MessageData-ChunkCount: 1 X-MS-Exchange-AntiSpam-MessageData-0: nZ82z/ZfJAeTqZ8uuRhVF1j/wZG0n4OQxuQLh3C3IGVAU+9wOW4zk8scA7kTYZwuggd2YO6/73aU8t8EdN6PUq0udGBKOymRnMlSrP2BNpw6pUjPJAwVtnHU1QB1b+RGozSqatN6bYvgNxW5xomQ+A0fPtIhEhRu4GLMzlNMRGtmSHqSDW9FewG140lJbO4RIL/5ghCkZ/OmEJHP27SRmz4NK7bE9jqB7nGvmBLyNDGs91tJuUGxvDorzvgZFKH37pwIN6nvOeh6lbLEihfEG2QVdsvJVDh52Czpy4IaRg2cvcgtTE8wGDpY3gZxqV0zlduW2yNvUEK/3IhyFCqatozuMJF2goxisvMfSo+cIYEdilw7Z2NcHIj2OXZI4SlvUpXWcXROvh8nMmRd9Nas4+GPnUe/rLwSYxFSpMXegKi+zH6BvYbuFO0bhkRPsMHPZRIWi3O7hnyWTYFOLlNqu6nYnf6O0W1flKDSqTQjn5gJtsq+99n3IhZ+VCKclKiOLPym+T2gk4LdFGl7mLrG54mTPahYLOIg/aKMMM7EK/3eNrK1muwiCIvvkRU+GgYqVswTa4WWyK2ML5s4tX0+3H/Uaeng+RGrSWX7KTrauNC1Do6dsNLTXtLKtluSQoCyg/37pBlcnoWtrT8qAR35lNRatT20AXW///7/ULAkB1s30MOPrjBevWtAWrE1trBgUeNW7y53ZOh2+LBx+chhkUPBsTMvAXkHO+Wmohfc/zVKcD0NSsasmvZRr1y1O8Kxfb+MIehsSx5suSrI9vIl2q0Yp2I4KxM7rmOM/dUXoMWX4VGJl0tZ8TbY8AJG34SuYWwpdUpyzRpiYmWuaLezAbveeY8TflNR91S1NGhjvPQaBmTNLpzhdMbEYdrh+hHdVgUz9toKoAznqKvqovsBvMc2MJ2NRVRt5Aodq8WXvI1Eia1vitOF10sI3tP0HEj1S1lYTv7QJ/u3qiYzWo/nFWXXaoXdv0qMfd7GZNIDmkjozNkfnRqAwRWxy07qlF3P5UT/TVB3HNL/L5C4frxJ1i/RLGP3M1HykZWY4og2Z6LuL2+EY5ZUL4/mraX9KxOknPQwvFVwmLUT9LLpwPrZ4GFnz+fOR7qbErLLjZLk8RZKcii24q/nlPcnQatoowExDAPmCh/w+pFHQOajRnqsd/KwRErW9RQy5D6vUkrfOm8f7eoF8XOsh13/778ttLU6QGJwZJFC+keZatOS5FVSCal+QGb2g97I2JvlOGo3fl0= X-OriginatorOrg: sct-15-20-4755-11-msonline-outlook-3208f.templateTenant X-MS-Exchange-CrossTenant-Network-Message-Id: 8720ca03-9047-43de-fbbc-08dc5302a933 X-MS-Exchange-CrossTenant-AuthSource: TY0PR06MB5611.apcprd06.prod.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Internal X-MS-Exchange-CrossTenant-OriginalArrivalTime: 02 Apr 2024 10:50:07.7753 (UTC) X-MS-Exchange-CrossTenant-FromEntityHeader: Hosted X-MS-Exchange-CrossTenant-Id: 84df9e7f-e9f6-40af-b435-aaaaaaaaaaaa X-MS-Exchange-CrossTenant-RMS-PersistedConsumerOrg: 00000000-0000-0000-0000-000000000000 X-MS-Exchange-Transport-CrossTenantHeadersStamped: KL1PR06MB6964 Add a new driver for Himax HX83102J touchscreen controllers. This driver supports Himax IC using the SPI interface to acquire HID packets. After confirmed the IC's exsitence the driver loads the firmware image from flash to get the HID report descriptor, VID and PID. And use those information to register HID device. Signed-off-by: Allen_Lin --- MAINTAINERS | 1 + drivers/hid/Kconfig | 7 + drivers/hid/Makefile | 2 + drivers/hid/hid-himax-83102j.c | 1695 ++++++++++++++++++++++++++++++++ drivers/hid/hid-himax-83102j.h | 286 ++++++ 5 files changed, 1991 insertions(+) create mode 100644 drivers/hid/hid-himax-83102j.c create mode 100644 drivers/hid/hid-himax-83102j.h diff --git a/MAINTAINERS b/MAINTAINERS index aa51c60fd66d..200db95b501b 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -9674,6 +9674,7 @@ M: Allen Lin L: linux-input@vger.kernel.org S: Maintained F: Documentation/devicetree/bindings/input/touchscreen/himax,hx83102j.yaml +F: drivers/hid/hid-himax-83102j.* HIMAX HX83112B TOUCHSCREEN SUPPORT M: Job Noorman diff --git a/drivers/hid/Kconfig b/drivers/hid/Kconfig index 4c682c650704..94eacf63da21 100644 --- a/drivers/hid/Kconfig +++ b/drivers/hid/Kconfig @@ -376,6 +376,13 @@ config HID_GLORIOUS Support for Glorious PC Gaming Race mice such as the Glorious Model O, O- and D. +config HID_HIMAX_HX83102J + tristate "Himax hx83102j touchscreen" + depends on HID + depends on SPI + help + Support for Himax TDDI hx83102j touchscreen. + config HID_HOLTEK tristate "Holtek HID devices" depends on USB_HID diff --git a/drivers/hid/Makefile b/drivers/hid/Makefile index 082a728eac60..0f6cb9ccb093 100644 --- a/drivers/hid/Makefile +++ b/drivers/hid/Makefile @@ -170,3 +170,5 @@ obj-$(INTEL_ISH_FIRMWARE_DOWNLOADER) += intel-ish-hid/ obj-$(CONFIG_AMD_SFH_HID) += amd-sfh-hid/ obj-$(CONFIG_SURFACE_HID_CORE) += surface-hid/ + +obj-$(CONFIG_HID_HIMAX_HX83102J) += hid-himax-83102j.o diff --git a/drivers/hid/hid-himax-83102j.c b/drivers/hid/hid-himax-83102j.c new file mode 100644 index 000000000000..aa8d0b6677bb --- /dev/null +++ b/drivers/hid/hid-himax-83102j.c @@ -0,0 +1,1695 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Himax hx83102j SPI Driver Code for HID. + * + * Copyright (C) 2024 Himax Corporation. + */ + +#include "hid-himax-83102j.h" + +static int himax_chip_init(struct himax_ts_data *ts); +static void himax_ts_work(struct himax_ts_data *ts); + +/** + * himax_spi_read() - Read data from SPI + * @ts: Himax touch screen data + * @cmd_len: Length of command + * @buf: Buffer to store data + * @len: Length of data to read + * + * Himax spi_sync wrapper for read. Read protocol start with write command, + * and received the data after that. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_spi_read(struct himax_ts_data *ts, u8 cmd_len, u8 *buf, u32 len) +{ + int ret; + int retry_cnt; + struct spi_message msg; + struct spi_transfer xfer = { + .len = cmd_len + len, + .tx_buf = ts->xfer_tx_data, + .rx_buf = ts->xfer_rx_data + }; + + spi_message_init(&msg); + spi_message_add_tail(&xfer, &msg); + + for (retry_cnt = 0; retry_cnt < HIMAX_BUS_RETRY; retry_cnt++) { + ret = spi_sync(ts->spi, &msg); + if (!ret) + break; + } + + if (retry_cnt == HIMAX_BUS_RETRY) { + dev_err(ts->dev, "%s: SPI read error retry over %d\n", __func__, HIMAX_BUS_RETRY); + return -EIO; + } + + if (ret < 0) + return ret; + + if (msg.status < 0) + return msg.status; + + memcpy(buf, ts->xfer_rx_data + cmd_len, len); + + return 0; +} + +/** + * himax_spi_write() - Write data to SPI + * @ts: Himax touch screen data + * @tx_buf: Buffer to write + * @tx_len: Length of data to write + * @written: Length of data written + * + * Himax spi_sync wrapper for write. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_spi_write(struct himax_ts_data *ts, u8 *tx_buf, u32 tx_len, u32 *written) +{ + int ret; + struct spi_message msg; + struct spi_transfer xfer = { + .tx_buf = tx_buf, + .len = tx_len, + }; + + spi_message_init(&msg); + spi_message_add_tail(&xfer, &msg); + + *written = 0; + ret = spi_sync(ts->spi, &msg); + + if (ret < 0) + return ret; + + if (msg.status < 0) + return msg.status; + + *written = msg.actual_length; + + return 0; +} + +/** + * himax_read() - Read data from Himax bus + * @ts: Himax touch screen data + * @cmd: Command to send + * @buf: Buffer to store data, caller should allocate the buffer + * @len: Length of data to read + * + * Basic read operation for Himax SPI bus. Which start with a 3 bytes command, + * 1st byte is the spi function select, 2nd byte is the command belong to the + * spi function and 3rd byte is the dummy byte for IC to process the command. + * + * The IC takes 1 basic operation at a time, so the read/write operation + * is proctected by rw_lock mutex_unlock. Also the buffer xfer_rx/tx_data is + * shared between read and write operation, protected by the same mutex lock. + * The xfer data limit by SPI constroller max xfer size + BUS_R/W_HLEN + * + * Return: 0 on success, negative error code on failure + */ +static int himax_read(struct himax_ts_data *ts, u8 cmd, u8 *buf, u32 len) +{ + int ret; + + if (len + HIMAX_BUS_R_HLEN > ts->spi_xfer_max_sz) { + dev_err(ts->dev, "%s, len[%u] is over %u\n", __func__, + len + HIMAX_BUS_R_HLEN, ts->spi_xfer_max_sz); + return -EINVAL; + } + + mutex_lock(&ts->rw_lock); + + memset(ts->xfer_rx_data, 0, HIMAX_BUS_R_HLEN + len); + ts->xfer_tx_data[0] = HIMAX_SPI_FUNCTION_READ; + ts->xfer_tx_data[1] = cmd; + ts->xfer_tx_data[2] = 0x00; + ret = himax_spi_read(ts, HIMAX_BUS_R_HLEN, buf, len); + + mutex_unlock(&ts->rw_lock); + if (ret < 0) + dev_err(ts->dev, "%s: failed = %d\n", __func__, ret); + + return ret; +} + +/** + * himax_write() - Write data to Himax bus + * @ts: Himax touch screen data + * @cmd: Command to send + * @addr: Address to write + * @data: Data to write + * @len: Length of data to write + * + * Basic write operation for Himax IC. Which start with a 2 bytes command, + * 1st byte is the spi function select and 2nd byte is the command belong to the + * spi function. Else is the data to write. + * + * The IC takes 1 basic operation at a time, so the read/write operation + * is proctected by rw_lock mutex_unlock. Also the buffer xfer_tx_data is + * shared between read and write operation, protected by the same mutex lock. + * The xfer data limit by SPI constroller max xfer size + HIMAX_BUS_W_HLEN + * + * Return: 0 on success, negative error code on failure + */ +static int himax_write(struct himax_ts_data *ts, u8 cmd, u8 *addr, const u8 *data, u32 len) +{ + int ret; + u8 offset; + u32 written; + u32 tmp_len; + + if (len + HIMAX_BUS_W_HLEN > ts->spi_xfer_max_sz) { + dev_err(ts->dev, "%s: len[%u] is over %u\n", __func__, + len + HIMAX_BUS_W_HLEN, ts->spi_xfer_max_sz); + return -EFAULT; + } + + mutex_lock(&ts->rw_lock); + + memset(ts->xfer_tx_data, 0, len + HIMAX_BUS_W_HLEN); + ts->xfer_tx_data[0] = HIMAX_SPI_FUNCTION_WRITE; + ts->xfer_tx_data[1] = cmd; + offset = HIMAX_BUS_W_HLEN; + tmp_len = len; + + if (addr) { + memcpy(ts->xfer_tx_data + offset, addr, 4); + offset += 4; + tmp_len -= 4; + } + + if (data) + memcpy(ts->xfer_tx_data + offset, data, tmp_len); + + ret = himax_spi_write(ts, ts->xfer_tx_data, len + HIMAX_BUS_W_HLEN, &written); + + mutex_unlock(&ts->rw_lock); + + if (ret < 0) { + dev_err(ts->dev, "%s: failed, ret = %d\n", __func__, ret); + return ret; + } + + if (written != len + HIMAX_BUS_W_HLEN) { + dev_err(ts->dev, "%s: actual write length mismatched: %u != %u\n", + __func__, written, len + HIMAX_BUS_W_HLEN); + return -EIO; + } + + return 0; +} + +/** + * himax_mcu_set_burst_mode() - Set burst mode for MCU + * @ts: Himax touch screen data + * @auto_add_4_byte: Enable auto add 4 byte mode + * + * Set burst mode for MCU, which is used for read/write data from/to MCU. + * HIMAX_AHB_ADDR_CONTI config the IC to take data continuously, + * HIMAX_AHB_ADDR_INCR4 config the IC to auto increment the address by 4 byte when + * each 4 bytes read/write. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_set_burst_mode(struct himax_ts_data *ts, bool auto_add_4_byte) +{ + int ret; + u8 tmp_data[HIMAX_REG_SZ]; + + tmp_data[0] = HIMAX_AHB_CMD_CONTI; + + ret = himax_write(ts, HIMAX_AHB_ADDR_CONTI, NULL, tmp_data, 1); + if (ret < 0) { + dev_err(ts->dev, "%s: write ahb_addr_conti failed\n", __func__); + return ret; + } + + tmp_data[0] = HIMAX_AHB_CMD_INCR4; + if (auto_add_4_byte) + tmp_data[0] |= HIMAX_AHB_CMD_INCR4_ADD_4_BYTE; + + ret = himax_write(ts, HIMAX_AHB_ADDR_INCR4, NULL, tmp_data, 1); + if (ret < 0) + dev_err(ts->dev, "%s: write ahb_addr_incr4 failed\n", __func__); + + return ret; +} + +/** + * himax_burst_mode_enable() - Enable burst mode for MCU if possible + * @ts: Himax touch screen data + * @addr: Address to read/write + * @len: Length of data to read/write + * + * Enable burst mode for MCU, helper function to determine the burst mode + * operation for MCU. When the address is HIMAX_REG_ADDR_SPI200_DATA, the burst + * mode is disabled. When the length of data is over HIMAX_REG_SZ, the burst + * mode is enabled. Else the burst mode is disabled. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_burst_mode_enable(struct himax_ts_data *ts, u32 addr, u32 len) +{ + int ret; + + if (addr == HIMAX_REG_ADDR_SPI200_DATA) + ret = himax_mcu_set_burst_mode(ts, false); + else if (len > HIMAX_REG_SZ) + ret = himax_mcu_set_burst_mode(ts, true); + else + ret = himax_mcu_set_burst_mode(ts, false); + + if (ret) + dev_err(ts->dev, "%s: burst enable fail!\n", __func__); + + return ret; +} + +/** + * himax_mcu_register_read() - Read data from IC register/sram + * @ts: Himax touch screen data + * @addr: Address to read + * @buf: Buffer to store data, caller should allocate the buffer + * @len: Length of data to read + * + * Himax TP IC has its internal register and SRAM, this function is used to + * read data from it. The reading protocol require a sequence of write and read, + * which include write address to IC and read data from IC. Thus the read/write + * operation is proctected by reg_lock mutex_unlock to protect the sequence. + * The first step is to set the burst mode for MCU, then write the address to + * AHB register to tell where to read. Then set the access direction to read, + * and read the data from AHB register. The max length of data to read is decided + * by AHB register max transfer size, but if it could't bigger then SPI controller + * max transfer size. When the length of data is over the max transfer size, + * the data will be read in multiple times. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_register_read(struct himax_ts_data *ts, u32 addr, u8 *buf, u32 len) +{ + int i; + int ret; + u8 direction_switch = HIMAX_AHB_CMD_ACCESS_DIRECTION_READ; + u32 read_sz; + const u32 max_trans_sz = + min(HIMAX_HX83102J_REG_XFER_MAX, ts->spi_xfer_max_sz - HIMAX_BUS_R_HLEN); + union himax_dword_data target_addr; + + mutex_lock(&ts->reg_lock); + + ret = himax_burst_mode_enable(ts, addr, len); + if (ret) + goto read_end; + + for (i = 0; i < len; i += read_sz) { + target_addr.dword = cpu_to_le32(addr + i); + ret = himax_write(ts, HIMAX_AHB_ADDR_BYTE_0, target_addr.byte, NULL, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: write ahb_addr_byte_0 failed\n", __func__); + goto read_end; + } + + ret = himax_write(ts, HIMAX_AHB_ADDR_ACCESS_DIRECTION, NULL, + &direction_switch, 1); + if (ret < 0) { + dev_err(ts->dev, "%s: write ahb_addr_access_direction failed\n", __func__); + goto read_end; + } + + read_sz = min((len - i), max_trans_sz); + ret = himax_read(ts, HIMAX_AHB_ADDR_RDATA_BYTE_0, buf + i, read_sz); + if (ret < 0) { + dev_err(ts->dev, "%s: read ahb_addr_rdata_byte_0 failed\n", __func__); + goto read_end; + } + } + +read_end: + mutex_unlock(&ts->reg_lock); + if (ret < 0) + dev_err(ts->dev, "%s: addr = 0x%08X, len = %u, ret = %d\n", __func__, + addr, len, ret); + + return ret; +} + +/** + * himax_mcu_register_write() - Write data to IC register/sram + * @ts: Himax touch screen data + * @addr: Address to write + * @buf: Data to write + * @len: Length of data to write + * + * Himax TP IC has its internal register and SRAM, this function is used to + * write data to it. The writing protocol require a sequence of write, which + * include write address to IC and write data to IC. Thus the write operation + * is proctected by reg_lock mutex_unlock to protect the sequence. The first + * step is to set the burst mode for MCU, then write the address and data to + * AHB register. The max length of data to read is decided by AHB register max + * transfer size, but if it could't bigger then SPI controller max transfer + * size. When the length of data is over the max transfer size, the data will + * be written in multiple times. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_register_write(struct himax_ts_data *ts, u32 addr, const u8 *buf, u32 len) +{ + int i; + int ret; + u32 write_sz; + const u32 max_trans_sz = min(HIMAX_HX83102J_REG_XFER_MAX, + ts->spi_xfer_max_sz - HIMAX_BUS_W_HLEN - HIMAX_REG_SZ); + union himax_dword_data target_addr; + + mutex_lock(&ts->reg_lock); + + ret = himax_burst_mode_enable(ts, addr, len); + if (ret) + goto write_end; + + for (i = 0; i < len; i += max_trans_sz) { + write_sz = min((len - i), max_trans_sz); + target_addr.dword = cpu_to_le32(addr + i); + ret = himax_write(ts, HIMAX_AHB_ADDR_BYTE_0, + target_addr.byte, buf + i, write_sz + HIMAX_REG_SZ); + if (ret < 0) { + dev_err(ts->dev, "%s: write ahb_addr_byte_0 failed\n", __func__); + break; + } + } + +write_end: + mutex_unlock(&ts->reg_lock); + if (ret < 0) + dev_err(ts->dev, "%s: addr = 0x%08X, len = %u, ret = %d\n", __func__, + addr, len, ret); + + return ret; +} + +/** + * himax_mcu_interface_on() - Wakeup IC bus interface + * @ts: Himax touch screen data + * + * This function is used to wakeup IC bus interface. The IC may enter sleep mode + * and need to wakeup before any operation. The wakeup process is to read a dummy + * AHB register to wakeup the IC bus interface. Also, the function setup the burst + * mode as default for MCU and read back the burst mode setting to confirm the + * setting is written. The action is a double check to confirm the IC bus interface + * is ready for operation. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_interface_on(struct himax_ts_data *ts) +{ + int ret; + u8 buf[2][HIMAX_REG_SZ]; + u32 retry_cnt; + const u32 burst_retry_limit = 10; + + mutex_lock(&ts->reg_lock); + /* Read a dummy register to wake up BUS. */ + ret = himax_read(ts, HIMAX_AHB_ADDR_RDATA_BYTE_0, buf[0], 4); + mutex_unlock(&ts->reg_lock); + if (ret < 0) { + dev_err(ts->dev, "%s: read ahb_addr_rdata_byte_0 failed\n", __func__); + return ret; + } + + for (retry_cnt = 0; retry_cnt < burst_retry_limit; retry_cnt++) { + /* AHB: read/write to SRAM in sequential order */ + buf[0][0] = HIMAX_AHB_CMD_CONTI; + ret = himax_write(ts, HIMAX_AHB_ADDR_CONTI, NULL, buf[0], 1); + if (ret < 0) { + dev_err(ts->dev, "%s: write ahb_addr_conti failed\n", __func__); + return ret; + } + + /* AHB: Auto increment SRAM addr+4 while each 4 bytes read/write */ + buf[0][0] = HIMAX_AHB_CMD_INCR4; + ret = himax_write(ts, HIMAX_AHB_ADDR_INCR4, NULL, buf[0], 1); + if (ret < 0) { + dev_err(ts->dev, "%s: write ahb_addr_incr4 failed\n", __func__); + return ret; + } + + /* Check cmd */ + ret = himax_read(ts, HIMAX_AHB_ADDR_CONTI, buf[0], 1); + if (ret < 0) { + dev_err(ts->dev, "%s: read ahb_addr_conti failed\n", __func__); + return ret; + } + + ret = himax_read(ts, HIMAX_AHB_ADDR_INCR4, buf[1], 1); + if (ret < 0) { + dev_err(ts->dev, "%s: read ahb_addr_incr4 failed\n", __func__); + return ret; + } + + if (buf[0][0] == HIMAX_AHB_CMD_CONTI && buf[1][0] == HIMAX_AHB_CMD_INCR4) + return 0; + + usleep_range(1000, 1100); + } + + dev_err(ts->dev, "%s: failed!\n", __func__); + + return -EIO; +} + +/** + * hx83102j_pin_reset() - Reset the touch chip by hardware pin + * @ts: Himax touch screen data + * + * This function is used to hardware reset the touch chip. By pull down the + * reset pin to low over 20ms, ensure the reset circuit perform a complete reset + * to the touch chip. + * + * Return: None + */ +static void hx83102j_pin_reset(struct himax_ts_data *ts) +{ + gpiod_set_value(ts->pdata.gpiod_rst, 1); + usleep_range(10000, 10100); + gpiod_set_value(ts->pdata.gpiod_rst, 0); + usleep_range(20000, 20100); +} + +/** + * himax_int_enable() - Enable/Disable interrupt + * @ts: Himax touch screen data + * @enable: true for enable, false for disable + * + * This function is used to enable or disable the interrupt. + * + * Return: None + */ +static void himax_int_enable(struct himax_ts_data *ts, bool enable) +{ + int irqnum = ts->himax_irq; + unsigned long flags; + + spin_lock_irqsave(&ts->irq_lock, flags); + if (enable && atomic_read(&ts->irq_state) == 0) { + atomic_set(&ts->irq_state, 1); + enable_irq(irqnum); + } else if (!enable && atomic_read(&ts->irq_state) == 1) { + atomic_set(&ts->irq_state, 0); + disable_irq_nosync(irqnum); + } + spin_unlock_irqrestore(&ts->irq_lock, flags); + dev_info(ts->dev, "%s: Interrupt %s\n", __func__, + atomic_read(&ts->irq_state) ? "enabled" : "disabled"); +} + +/** + * himax_mcu_ic_reset() - Reset the touch chip and disable/enable interrupt + * @ts: Himax touch screen data + * @int_off: true for disable/enable interrupt, false for not + * + * This function is used to reset the touch chip with interrupt control. The + * TPIC will pull low the interrupt pin when IC is reset. When the ISR has been + * set and need to be take care of, the caller could set int_off to true to disable + * the interrupt before reset and enable the interrupt after reset. + * + * Return: None + */ +static void himax_mcu_ic_reset(struct himax_ts_data *ts, bool int_off) +{ + if (int_off) + himax_int_enable(ts, false); + + hx83102j_pin_reset(ts); + + if (int_off) + himax_int_enable(ts, true); +} + +/** + * hx83102j_sense_off() - Stop MCU and enter safe mode + * @ts: Himax touch screen data + * @check_en: Check if need to ensure FW is stopped by its owne process + * + * Sense off is a process to make sure the MCU inside the touch chip is stopped. + * The process has two stage, first stage is to request FW to stop. Write + * HIMAX_REG_DATA_FW_GO_SAFEMODE to HIMAX_REG_ADDR_CTRL_FW tells the FW to stop by its own. + * Then read back the FW status to confirm the FW is stopped. When check_en is true, + * the function will resend the stop FW command until the retry limit reached. + * There maybe a chance that the FW is not stopped by its own, in this case, the + * safe mode in next stage still stop the MCU, but FW internal flag may not be + * configured correctly. The second stage is to enter safe mode and reset TCON. + * Safe mode is a mode that the IC circuit ensure the internal MCU is stopped. + * Since this IC is TDDI, the TCON need to be reset to make sure the IC is ready + * for next operation. + * + * Return: 0 on success, negative error code on failure + */ +static int hx83102j_sense_off(struct himax_ts_data *ts, bool check_en) +{ + int ret; + u32 retry_cnt; + const u32 stop_fw_retry_limit = 35; + const u32 enter_safe_mode_retry_limit = 5; + const union himax_dword_data safe_mode = { + .dword = cpu_to_le32(HIMAX_REG_DATA_FW_GO_SAFEMODE) + }; + union himax_dword_data data; + + dev_info(ts->dev, "%s: check %s\n", __func__, check_en ? "True" : "False"); + if (!check_en) + goto without_check; + + for (retry_cnt = 0; retry_cnt < stop_fw_retry_limit; retry_cnt++) { + if (retry_cnt == 0 || + (data.byte[0] != HIMAX_REG_DATA_FW_GO_SAFEMODE && + data.byte[0] != HIMAX_REG_DATA_FW_RE_INIT && + data.byte[0] != HIMAX_REG_DATA_FW_IN_SAFEMODE)) { + ret = himax_mcu_register_write(ts, HIMAX_REG_ADDR_CTRL_FW, + safe_mode.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: stop FW failed\n", __func__); + return ret; + } + } + usleep_range(10000, 11000); + + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_FW_STATUS, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read central state failed\n", __func__); + return ret; + } + if (data.byte[0] != HIMAX_REG_DATA_FW_STATE_RUNNING) { + dev_info(ts->dev, "%s: Do not need wait FW, Status = 0x%02X!\n", __func__, + data.byte[0]); + break; + } + + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_CTRL_FW, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read ctrl FW failed\n", __func__); + return ret; + } + if (data.byte[0] == HIMAX_REG_DATA_FW_IN_SAFEMODE) + break; + } + + if (data.byte[0] != HIMAX_REG_DATA_FW_IN_SAFEMODE) + dev_warn(ts->dev, "%s: Failed to stop FW!\n", __func__); + +without_check: + for (retry_cnt = 0; retry_cnt < enter_safe_mode_retry_limit; retry_cnt++) { + /* set Enter safe mode : 0x31 ==> 0x9527 */ + data.word[0] = cpu_to_le16(HIMAX_HX83102J_SAFE_MODE_PASSWORD); + ret = himax_write(ts, HIMAX_AHB_ADDR_PSW_LB, NULL, data.byte, 2); + if (ret < 0) { + dev_err(ts->dev, "%s: enter safe mode failed\n", __func__); + return ret; + } + + /* Check enter_save_mode */ + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_FW_STATUS, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read central state failed\n", __func__); + return ret; + } + + if (data.byte[0] == HIMAX_REG_DATA_FW_STATE_SAFE_MODE) { + dev_info(ts->dev, "%s: Safe mode entered\n", __func__); + /* Reset TCON */ + data.dword = cpu_to_le32(HIMAX_REG_DATA_TCON_RST); + ret = himax_mcu_register_write(ts, HIMAX_HX83102J_REG_ADDR_TCON_RST, + data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: reset TCON failed\n", __func__); + return ret; + } + usleep_range(1000, 1100); + return 0; + } + usleep_range(5000, 5100); + hx83102j_pin_reset(ts); + } + dev_err(ts->dev, "%s: failed!\n", __func__); + + return -EIO; +} + +/** + * hx83102j_chip_detect() - Check if the touch chip is HX83102J + * @ts: Himax touch screen data + * + * This function is used to check if the touch chip is HX83102J. The process + * start with a hardware reset to the touch chip, then knock the IC bus interface + * to wakeup the IC bus interface. Then sense off the MCU to prevent bus conflict + * when reading the IC ID. The IC ID is read from the IC register, and compare + * with the expected ID. If the ID is matched, the chip is HX83102J. Due to display + * IC initial code may not ready before the IC ID is read, the function will retry + * to read the IC ID for several times to make sure the IC ID is read correctly. + * In any case, the SPI bus shouldn't have error when reading the IC ID, so the + * function will return error if the SPI bus has error. When the IC is not HX83102J, + * the function will return -ENODEV. + * + * Return: 0 on success, negative error code on failure + */ +static int hx83102j_chip_detect(struct himax_ts_data *ts) +{ + int ret; + u32 retry_cnt; + const u32 read_icid_retry_limit = 5; + const u32 ic_id_mask = GENMASK(31, 8); + union himax_dword_data data; + + hx83102j_pin_reset(ts); + ret = himax_mcu_interface_on(ts); + if (ret) + return ret; + + ret = hx83102j_sense_off(ts, false); + if (ret) + return ret; + + for (retry_cnt = 0; retry_cnt < read_icid_retry_limit; retry_cnt++) { + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_ICID, data.byte, 4); + if (ret) { + dev_err(ts->dev, "%s: Read IC ID Fail\n", __func__); + return ret; + } + + data.dword = le32_to_cpu(data.dword); + if ((data.dword & ic_id_mask) == HIMAX_REG_DATA_ICID) { + ts->ic_data.icid = data.dword; + dev_info(ts->dev, "%s: Detect IC HX83102J successfully\n", __func__); + return 0; + } + } + dev_err(ts->dev, "%s: Read driver ID register Fail! IC ID = %X,%X,%X\n", __func__, + data.byte[3], data.byte[2], data.byte[1]); + + return -ENODEV; +} + +/** + * himax_ts_thread() - Thread for interrupt handling + * @irq: Interrupt number + * @ptr: Pointer to the touch screen data + * + * This function is used to handle the interrupt. The function will call himax_ts_work() + * to handle the interrupt. + * + * Return: IRQ_HANDLED + */ +static irqreturn_t himax_ts_thread(int irq, void *ptr) +{ + himax_ts_work((struct himax_ts_data *)ptr); + + return IRQ_HANDLED; +} + +/** + * __himax_ts_register_interrupt() - Register interrupt trigger + * @ts: Himax touch screen data + * + * This function is used to register the interrupt. The function will call + * devm_request_threaded_irq() to register the interrupt by the trigger type. + * + * Return: 0 on success, negative error code on failure + */ +static int __himax_ts_register_interrupt(struct himax_ts_data *ts) +{ + if (ts->ic_data.interrupt_is_edge) + return devm_request_threaded_irq(ts->dev, ts->himax_irq, NULL, + himax_ts_thread, + IRQF_TRIGGER_FALLING | IRQF_ONESHOT, + ts->dev->driver->name, ts); + + return devm_request_threaded_irq(ts->dev, ts->himax_irq, NULL, + himax_ts_thread, + IRQF_TRIGGER_LOW | IRQF_ONESHOT, + ts->dev->driver->name, ts); +} + +/** + * himax_ts_register_interrupt() - Register interrupt + * @ts: Himax touch screen data + * + * This function is a wrapper to call __himax_ts_register_interrupt() to register the + * interrupt and set irq_state. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_ts_register_interrupt(struct himax_ts_data *ts) +{ + int ret; + + if (!ts || !ts->himax_irq) { + dev_err(ts->dev, "%s: ts or ts->himax_irq invalid!\n", __func__); + return -EINVAL; + } + + ret = __himax_ts_register_interrupt(ts); + if (!ret) { + atomic_set(&ts->irq_state, 1); + dev_info(ts->dev, "%s: irq enabled at: %d\n", __func__, ts->himax_irq); + return 0; + } + + atomic_set(&ts->irq_state, 0); + dev_err(ts->dev, "%s: request_irq failed\n", __func__); + + return ret; +} + +/** + * hx83102j_read_event_stack() - Read event stack from touch chip + * @ts: Himax touch screen data + * @buf: Buffer to store the data + * @length: Length of data to read + * + * This function is used to read the event stack from the touch chip. The event stack + * is an AHB output buffer, which store the touch report data. + * + * Return: 0 on success, negative error code on failure + */ +static int hx83102j_read_event_stack(struct himax_ts_data *ts, u8 *buf, u32 length) +{ + u32 i; + int ret; + const u32 max_trunk_sz = ts->spi_xfer_max_sz - HIMAX_BUS_R_HLEN; + + for (i = 0; i < length; i += max_trunk_sz) { + ret = himax_read(ts, HIMAX_AHB_ADDR_EVENT_STACK, buf + i, + min(length - i, max_trunk_sz)); + if (ret) { + dev_err(ts->dev, "%s: read event stack error!\n", __func__); + return ret; + } + } + + return 0; +} + +/** + * himax_touch_get() - Get touch data from touch chip + * @ts: Himax touch screen data + * @buf: Buffer to store the data + * + * This function is a wrapper to call hx83102j_read_event_stack() to read the touch + * data from the touch chip. The touch_data_sz is the size of the touch data to read, + * which is calculated by hid report descriptor provided by the firmware. + * + * Return: HIMAX_TS_SUCCESS on success, negative error code on failure. We categorize + * the error code into HIMAX_TS_GET_DATA_FAIL when the read fails, and HIMAX_TS_SUCCESS + * when the read is successful. The reason is that the may need special handling when + * the read fails. + */ +static int himax_touch_get(struct himax_ts_data *ts, u8 *buf) +{ + if (hx83102j_read_event_stack(ts, buf, ts->touch_data_sz)) { + dev_err(ts->dev, "can't read data from chip!"); + return HIMAX_TS_GET_DATA_FAIL; + } + + return HIMAX_TS_SUCCESS; +} + +/** + * himax_bin_desc_data_get() - Parse descriptor data from firmware token + * @ts: Himax touch screen data + * @addr: Address of the data in firmware image + * @descript_buf: token for parsing + * + * This function is used to parse the descriptor data from the firmware token. The + * descriptors are mappings of information in the firmware image. The function will + * check checksum of each token first, and then parse the token to get the related + * data. The data includes CID version, FW version, CFG version, touch config table, + * HID table, HID descriptor, and HID read descriptor. + * + * Return: true on success, false on failure + */ +static bool himax_bin_desc_data_get(struct himax_ts_data *ts, u32 addr, u8 *descript_buf) +{ + u16 chk_end; + u16 chk_sum; + u32 hid_table_addr; + u32 i, j; + u32 image_offset; + u32 map_code; + const u32 data_sz = 16; + const u32 report_desc_offset = 24; + union { + u8 *buf; + u32 *word; + } map_data; + + /* looking for mapping in page, each mapping is 16 bytes */ + for (i = 0; i < HIMAX_HX83102J_PAGE_SIZE; i = i + data_sz) { + chk_end = 0; + chk_sum = 0; + for (j = i; j < (i + data_sz); j++) { + chk_end |= descript_buf[j]; + chk_sum += descript_buf[j]; + } + if (!chk_end) { /* 1. Check all zero */ + return false; + } else if (chk_sum % 0x100) { /* 2. Check sum */ + dev_warn(ts->dev, "%s: chk sum failed in %X\n", __func__, i + addr); + } else { /* 3. get data */ + map_data.buf = &descript_buf[i]; + map_code = le32_to_cpup(map_data.word); + map_data.buf = &descript_buf[i + 4]; + image_offset = le32_to_cpup(map_data.word); + /* 4. load info from FW image by specified mapping offset */ + switch (map_code) { + /* Config ID */ + case HIMAX_FW_CID: + ts->fw_info_table.addr_cid_ver_major = image_offset; + ts->fw_info_table.addr_cid_ver_minor = image_offset + 1; + break; + /* FW version */ + case HIMAX_FW_VER: + ts->fw_info_table.addr_fw_ver_major = image_offset; + ts->fw_info_table.addr_fw_ver_minor = image_offset + 1; + break; + /* Config version */ + case HIMAX_CFG_VER: + ts->fw_info_table.addr_cfg_ver_major = image_offset; + ts->fw_info_table.addr_cfg_ver_minor = image_offset + 1; + break; + /* Touch config table */ + case HIMAX_TP_CONFIG_TABLE: + ts->fw_info_table.addr_cfg_table = image_offset; + break; + /* HID table */ + case HIMAX_HID_TABLE: + ts->fw_info_table.addr_hid_table = image_offset; + hid_table_addr = image_offset; + ts->fw_info_table.addr_hid_desc = hid_table_addr; + ts->fw_info_table.addr_hid_rd_desc = + hid_table_addr + report_desc_offset; + break; + } + } + } + + return true; +} + +/** + * himax_mcu_bin_desc_get() - Check and get the bin description from the data + * @fw: Firmware data + * @ts: Himax touch screen data + * @max_sz: Maximum size to check + * + * This function is used to check and get the bin description from the firmware data. + * It will check the given data to see if it match the bin description format, and + * call himax_bin_desc_data_get() to get the related data. + * + * Return: true on mapping_count > 0, false on otherwise + */ +static bool himax_mcu_bin_desc_get(unsigned char *fw, struct himax_ts_data *ts, u32 max_sz) +{ + bool keep_on_flag; + u32 addr; + u32 mapping_count; + unsigned char *fw_buf; + const u8 header_id = 0x87; + const u8 header_id_loc = 0x0e; + const u8 header_sz = 8; + const u8 header[8] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 + }; + + /* Check bin is with description table or not */ + if (!(memcmp(fw, header, header_sz) == 0 && fw[header_id_loc] == header_id)) { + dev_err(ts->dev, "%s: No description table\n", __func__); + return false; + } + + for (addr = 0, mapping_count = 0; addr < max_sz; addr += HIMAX_HX83102J_PAGE_SIZE) { + fw_buf = &fw[addr]; + /* Get related data */ + keep_on_flag = himax_bin_desc_data_get(ts, addr, fw_buf); + if (keep_on_flag) + mapping_count++; + else + break; + } + + return mapping_count > 0; +} + +/** + * himax_hid_parse() - Parse the HID report descriptor + * @hid: HID device + * + * This function is used to parse the HID report descriptor. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_hid_parse(struct hid_device *hid) +{ + int ret; + struct himax_ts_data *ts; + + if (!hid) + return -ENODEV; + + ts = hid->driver_data; + if (!ts) + return -EINVAL; + + ret = hid_parse_report(hid, ts->hid_rd_data.rd_data, ts->hid_rd_data.rd_length); + if (ret) { + dev_err(ts->dev, "%s: failed parse report\n", __func__); + return ret; + } + + return 0; +} + +/** + * himax_hid_start - Start the HID device + * @hid: HID device + * + * The function for hid_ll_driver.start to start the HID device. + * This driver does not need to do anything here. + * + * Return: 0 for success + */ +static int himax_hid_start(struct hid_device *hid) +{ + return 0; +} + +/** + * himax_hid_stop - Stop the HID device + * @hid: HID device + * + * The function for hid_ll_driver.stop to stop the HID device. + * This driver does not need to do anything here. + * + * Return: None + */ +static void himax_hid_stop(struct hid_device *hid) +{ +} + +/** + * himax_hid_open - Open the HID device + * @hid: HID device + * + * The function for hid_ll_driver.open to open the HID device. + * This driver does not need to do anything here. + * + * Return: 0 for success + */ +static int himax_hid_open(struct hid_device *hid) +{ + return 0; +} + +/** + * himax_hid_close - Close the HID device + * @hid: HID device + * + * The function for hid_ll_driver.close to close the HID device. + * This driver does not need to do anything here. + * + * Return: None + */ +static void himax_hid_close(struct hid_device *hid) +{ +} + +/** + * himax_hid_get_raw_report - Process hidraw GET REPORT operation + * @hid: HID device + * @reportnum: Report ID + * @buf: Buffer for communication + * @len: Length of data in the buffer + * @report_type: Report type + * + * The function for hid_ll_driver.get_raw_report to handle the HIDRAW ioctl + * get report request. The report number to handle is based on the report + * descriptor of the HID device. The buf is used to communicate with user + * program, user pass the ID and parameters to the driver use this buf, and + * the driver will return the result to user also use this buf. The len is + * the length of data in the buf, passed by user program. The report_type is + * not used in this driver. We currently support the following report number: + * - HIMAX_ID_CONTACT_COUNT: Report the maximum number of touch points + * Case not listed here will return -EINVAL. + * + * Return: The length of the data in the buf on success, negative error code + */ +static int himax_hid_get_raw_report(const struct hid_device *hid, + unsigned char reportnum, __u8 *buf, + size_t len, unsigned char report_type) +{ + struct himax_ts_data *ts; + + ts = hid->driver_data; + if (!ts) { + dev_err(ts->dev, "hid->driver_data is NULL"); + return -EINVAL; + } + + switch (reportnum) { + case HIMAX_ID_CONTACT_COUNT: + /* buf[0] is ID, keep it; buf[1] and later used as parameters for ID */ + buf[1] = ts->ic_data.max_point; + return len; + }; + + return -EINVAL; +} + +/** + * himax_raw_request - Handle the HIDRAW ioctl request + * @hid: HID device + * @reportnum: Report ID + * @buf: Buffer for communication + * @len: Length of data in the buffer + * @rtype: Report type + * @reqtype: Request type + * + * The function for hid_ll_driver.raw_request to handle the HIDRAW ioctl + * request. We handle only the GET_REPORT and SET_REPORT request. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_raw_request(struct hid_device *hid, unsigned char reportnum, __u8 *buf, + size_t len, unsigned char rtype, int reqtype) +{ + switch (reqtype) { + case HID_REQ_GET_REPORT: + return himax_hid_get_raw_report(hid, reportnum, buf, len, rtype); + default: + return -EIO; + } + + return -EINVAL; +} + +static struct hid_ll_driver himax_hid_ll_driver = { + .parse = himax_hid_parse, + .start = himax_hid_start, + .stop = himax_hid_stop, + .open = himax_hid_open, + .close = himax_hid_close, + .raw_request = himax_raw_request +}; + +/** + * himax_hid_report() - Report input data to HID core + * @ts: Himax touch screen data + * @data: Data to report + * @len: Length of the data + * + * This function is a wrapper to report input data to HID core. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_hid_report(const struct himax_ts_data *ts, u8 *data, s32 len) +{ + return hid_input_report(ts->hid, HID_INPUT_REPORT, data, len, 1); +} + +/** + * himax_hid_probe() - Probe the HID device + * @ts: Himax touch screen data + * + * This function is used to probe the HID device. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_hid_probe(struct himax_ts_data *ts) +{ + int ret; + struct hid_device *hid; + + hid = ts->hid; + if (hid) { + dev_warn(ts->dev, "%s: hid device already exist!\n", __func__); + hid_destroy_device(hid); + hid = NULL; + } + + hid = hid_allocate_device(); + if (IS_ERR(hid)) { + ret = PTR_ERR(hid); + return ret; + } + + hid->driver_data = ts; + hid->ll_driver = &himax_hid_ll_driver; + hid->bus = BUS_SPI; + hid->dev.parent = &ts->spi->dev; + + hid->version = ts->hid_desc.bcd_version; + hid->vendor = ts->hid_desc.vendor_id; + hid->product = ts->hid_desc.product_id; + snprintf(hid->name, sizeof(hid->name), "%s %04X:%04X", "hid-hxtp", + hid->vendor, hid->product); + + ret = hid_add_device(hid); + if (ret) { + dev_err(ts->dev, "%s: failed add hid device\n", __func__); + goto err_hid_data; + } + ts->hid = hid; + + return 0; + +err_hid_data: + hid_destroy_device(hid); + + return ret; +} + +/** + * himax_hid_remove() - Remove the HID device + * @ts: Himax touch screen data + * + * This function is used to remove the HID device. + * + * Return: None + */ +static void himax_hid_remove(struct himax_ts_data *ts) +{ + if (ts && ts->hid) + hid_destroy_device(ts->hid); + else + return; + + ts->hid = NULL; +} + +/** + * himax_ts_operation() - Process the touch interrupt data + * @ts: Himax touch screen data + * + * This function is used to process the touch interrupt data. It will + * call the himax_touch_get() to get the touch data. + * If the hid is probed, it will call the himax_hid_report() to report the + * touch data to the HID core. Due to the report data must match the HID + * report descriptor, the size of report data is fixed. To prevent next report + * data being contaminated by the previous data, all the data must be reported + * wheather previous data is valid or not. + * + * Return: HIMAX_TS_SUCCESS on success, negative error code in + * himax_touch_report_status on failure + */ +static int himax_ts_operation(struct himax_ts_data *ts) +{ + int ret; + u32 offset; + + memset(ts->xfer_buf, 0x00, ts->xfer_buf_sz); + ret = himax_touch_get(ts, ts->xfer_buf); + if (ret == HIMAX_TS_GET_DATA_FAIL) + return ret; + if (ts->hid_probed) { + offset = ts->hid_desc.max_input_length; + if (ts->ic_data.stylus_function) { + ret += himax_hid_report(ts, + ts->xfer_buf + offset + HIMAX_HID_REPORT_HDR_SZ, + ts->hid_desc.max_input_length - + HIMAX_HID_REPORT_HDR_SZ); + offset += ts->hid_desc.max_input_length; + } + } + + if (ret != 0) + return HIMAX_TS_GET_DATA_FAIL; + + return HIMAX_TS_SUCCESS; +} + +/** + * himax_ts_work() - Work function for the touch screen + * @ts: Himax touch screen data + * + * This function is used to handle interrupt bottom half work. It will + * call the himax_ts_operation() to get the touch data, dispatch the data + * to HID core. If the touch data is not valid, it will reset the TPIC. + * + * Return: void + */ +static void himax_ts_work(struct himax_ts_data *ts) +{ + if (himax_ts_operation(ts) == HIMAX_TS_GET_DATA_FAIL) { + dev_info(ts->dev, "%s: Now reset the Touch chip\n", __func__); + himax_mcu_ic_reset(ts, true); + } +} + +/** + * himax_hid_rd_init() - Initialize the HID report descriptor + * @ts: Himax touch screen data + * + * The function is used to calculate the size of the HID report descriptor, + * allocate the memory and copy the report descriptor from firmware data to + * the allocated memory for later hid device registration. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_hid_rd_init(struct himax_ts_data *ts) +{ + u32 rd_sz; + + /* The rd_sz is taken from RD size in FW hid report table. */ + rd_sz = ts->hid_desc.report_desc_length; + /* fw_info_table should contain address of hid_rd_desc in FW image */ + if (ts->fw_info_table.addr_hid_rd_desc != 0) { + /* if rd_sz has been change, need to release old one */ + if (ts->hid_rd_data.rd_data && + rd_sz != ts->hid_rd_data.rd_length) { + devm_kfree(ts->dev, ts->hid_rd_data.rd_data); + ts->hid_rd_data.rd_data = NULL; + } + + if (!ts->hid_rd_data.rd_data) { + ts->hid_rd_data.rd_data = devm_kzalloc(ts->dev, rd_sz, GFP_KERNEL); + if (!ts->hid_rd_data.rd_data) + return -ENOMEM; + } + /* Copy the base RD from firmware table */ + memcpy((void *)ts->hid_rd_data.rd_data, + &ts->himax_fw_data[ts->fw_info_table.addr_hid_rd_desc], + ts->hid_desc.report_desc_length); + ts->hid_rd_data.rd_length = ts->hid_desc.report_desc_length; + } + + return 0; +} + +/** + * himax_hid_register() - Register the HID device + * @ts: Himax touch screen data + * + * The function is used to register the HID device. If the hid is probed, + * it will destroy the previous hid device and re-probe the hid device. + * + * Return: None + */ +static void himax_hid_register(struct himax_ts_data *ts) +{ + if (ts->hid_probed) { + hid_destroy_device(ts->hid); + ts->hid = NULL; + ts->hid_probed = false; + } + + if (himax_hid_probe(ts)) { + dev_err(ts->dev, "%s: hid probe fail\n", __func__); + ts->hid_probed = false; + } else { + ts->hid_probed = true; + } +} + +/** + * himax_hid_report_data_init() - Calculate the size of the HID report data + * @ts: Himax touch screen data + * + * The function is used to calculate the final size of the HID report data. + * The base size is equal to the max_input_length of the hid descriptor. + * If the size of the HID report data is not equal to the previous size, it + * will free the previous allocated memory and allocate the new memory which + * size is equal to the final size of touch_data_sz. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_hid_report_data_init(struct himax_ts_data *ts) +{ + ts->touch_data_sz = ts->hid_desc.max_input_length; + if (ts->ic_data.stylus_function) + ts->touch_data_sz += ts->hid_desc.max_input_length; + if (ts->touch_data_sz != ts->xfer_buf_sz) { + kfree(ts->xfer_buf); + ts->xfer_buf_sz = 0; + ts->xfer_buf = kzalloc(ts->touch_data_sz, GFP_KERNEL); + if (!ts->xfer_buf) + return -ENOMEM; + ts->xfer_buf_sz = ts->touch_data_sz; + } + + return 0; +} + +/* load firmware data from flash, parse HID info and register HID */ +/** + * himax_load_config() - Load the firmware from the flash + * @ts: Himax touch screen data + * + * This function is used to load the firmware from the flash. It will read + * the firmware from the flash and parse the HID info. If the HID info is + * valid, it will initialize the HID report descriptor and register the HID + * device. If the HID device is probed, it will initialize the report data + * and enable the interrupt. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_load_config(struct himax_ts_data *ts) +{ + int ret; + s32 i; + s32 page_sz = (s32)HIMAX_HX83102J_PAGE_SIZE; + s32 flash_sz = (s32)HIMAX_HX83102J_FLASH_SIZE; + bool fw_load_status = false; + const u32 fw_bin_header_sz = 1024; + + ts->ic_boot_done = false; + + ts->himax_fw_data = devm_kzalloc(ts->dev, HIMAX_HX83102J_FLASH_SIZE, GFP_KERNEL); + if (!ts->himax_fw_data) + return -ENOMEM; + + for (i = 0; i < flash_sz; i += page_sz) { + ret = himax_mcu_register_read(ts, i, ts->himax_fw_data + i, + (flash_sz - i) > page_sz ? page_sz : (flash_sz - i)); + if (ret < 0) { + dev_err(ts->dev, "%s: read FW from flash fail!\n", __func__); + return ret; + } + } + /* Search mapping table in 1k header */ + fw_load_status = himax_mcu_bin_desc_get((unsigned char *)ts->himax_fw_data, + ts, fw_bin_header_sz); + if (!fw_load_status) { + dev_err(ts->dev, "%s: FW load status fail!\n", __func__); + return -EINVAL; + } + + if (ts->fw_info_table.addr_hid_desc != 0) { + memcpy(&ts->hid_desc, + &ts->himax_fw_data[ts->fw_info_table.addr_hid_desc], + sizeof(struct himax_hid_desc)); + ts->hid_desc.desc_length = + le16_to_cpu(ts->hid_desc.desc_length); + ts->hid_desc.bcd_version = + le16_to_cpu(ts->hid_desc.bcd_version); + ts->hid_desc.report_desc_length = + le16_to_cpu(ts->hid_desc.report_desc_length); + ts->hid_desc.max_input_length = + le16_to_cpu(ts->hid_desc.max_input_length); + ts->hid_desc.max_output_length = + le16_to_cpu(ts->hid_desc.max_output_length); + ts->hid_desc.max_fragment_length = + le16_to_cpu(ts->hid_desc.max_fragment_length); + ts->hid_desc.vendor_id = + le16_to_cpu(ts->hid_desc.vendor_id); + ts->hid_desc.product_id = + le16_to_cpu(ts->hid_desc.product_id); + ts->hid_desc.version_id = + le16_to_cpu(ts->hid_desc.version_id); + ts->hid_desc.flags = + le16_to_cpu(ts->hid_desc.flags); + } + + if (himax_hid_rd_init(ts)) { + dev_err(ts->dev, "%s: hid rd init fail\n", __func__); + goto err_hid_rd_init_failed; + } + + himax_hid_register(ts); + if (!ts->hid_probed) { + goto err_hid_probe_failed; + } else { + if (himax_hid_report_data_init(ts)) { + dev_err(ts->dev, "%s: report data init fail\n", __func__); + goto err_report_data_init_failed; + } + } + + ts->himax_fw_data = NULL; + ts->ic_boot_done = true; + himax_int_enable(ts, true); + + return 0; + +err_report_data_init_failed: + himax_hid_remove(ts); + ts->hid_probed = false; +err_hid_probe_failed: +err_hid_rd_init_failed: + + return -EINVAL; +} + +/** + * himax_chip_init() - Initialize the Himax touch screen + * @ts: Himax touch screen data + * + * This function is used to initialize the Himax touch screen. It will + * initialize interrupt lock, register the interrupt, and disable the + * interrupt. If later part of initialization succeed, then interrupt will + * be enabled. + * It will also load the firmware from the flash, parse the HID info, and + * register the HID device by calling the himax_load_config(). + * + * Return: 0 on success, negative error code on failure + */ +static int himax_chip_init(struct himax_ts_data *ts) +{ + int ret; + + if (himax_ts_register_interrupt(ts)) { + dev_err(ts->dev, "%s: register interrupt failed\n", __func__); + return -EIO; + } + himax_int_enable(ts, false); + ret = himax_load_config(ts); + if (ret < 0) + return ret; + ts->initialized = true; + + return 0; +} + +/** + * himax_platform_deinit() - Deinitialize the platform related settings + * @ts: Pointer to the himax_ts_data structure + * + * This function deinitializes the platform related settings, frees the + * xfer_buf. + * + * Return: None + */ +static void himax_platform_deinit(struct himax_ts_data *ts) +{ + if (ts->xfer_buf_sz) { + kfree(ts->xfer_buf); + ts->xfer_buf = NULL; + ts->xfer_buf_sz = 0; + } +} + +/** + * himax_platform_init - Initialize the platform related settings + * @ts: Pointer to the himax_ts_data structure + * + * This function initializes the platform related settings. + * The xfer_buf is used for interrupt data receive. gpio reset pin is set to + * active and then deactivate to reset the IC. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_platform_init(struct himax_ts_data *ts) +{ + ts->xfer_buf_sz = 0; + ts->xfer_buf = kzalloc(HIMAX_HX83102J_FULL_STACK_SZ, GFP_KERNEL); + if (!ts->xfer_buf) + return -ENOMEM; + ts->xfer_buf_sz = HIMAX_HX83102J_FULL_STACK_SZ; + + hx83102j_pin_reset(ts); + + return 0; +} + +/** + * himax_spi_drv_probe - Probe function for the SPI driver + * @spi: Pointer to the spi_device structure + * + * This function is called when the SPI driver is probed. It initializes the + * himax_ts_data structure and assign the settings from spi device to + * himax_ts_data. The buffer for SPI transfer is allocate here. The SPI + * transfer settings also setup before any communication starts. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_spi_drv_probe(struct spi_device *spi) +{ + int ret; + struct himax_ts_data *ts; + static struct himax_platform_data *pdata; + + dev_info(&spi->dev, "%s: Himax SPI driver probe\n", __func__); + ts = devm_kzalloc(&spi->dev, sizeof(struct himax_ts_data), GFP_KERNEL); + if (!ts) + return -ENOMEM; + if (spi->master->flags & SPI_MASTER_HALF_DUPLEX) { + dev_err(ts->dev, "%s: Full duplex not supported by host\n", __func__); + return -EIO; + } + pdata = &ts->pdata; + ts->dev = &spi->dev; + if (!spi->irq) { + dev_err(ts->dev, "%s: no IRQ?\n", __func__); + return -EINVAL; + } + ts->himax_irq = spi->irq; + pdata->gpiod_rst = devm_gpiod_get(ts->dev, "reset", GPIOD_OUT_HIGH); + if (IS_ERR(pdata->gpiod_rst)) { + dev_err(ts->dev, "%s: gpio-rst value is not valid\n", __func__); + return -EIO; + } + + spi->bits_per_word = 8; + spi->mode = SPI_MODE_3; + spi->cs_setup.value = HIMAX_SPI_CS_SETUP_TIME; + + ts->spi = spi; + /* + * The max_transfer_size is used to allocate the buffer for SPI transfer. + * The size should be given by the SPI master driver, but if not available + * then use the HIMAX_MAX_TP_EV_STACK_SZ as default. Which is the least size for + * each TP event data. + */ + if (spi->master->max_transfer_size) + ts->spi_xfer_max_sz = spi->master->max_transfer_size(spi); + else + ts->spi_xfer_max_sz = HIMAX_MAX_TP_EV_STACK_SZ; + + ts->spi_xfer_max_sz = min(ts->spi_xfer_max_sz, HIMAX_BUS_RW_MAX_LEN); + /* SPI full-duplex rx_buf and tx_buf should be equal */ + ts->xfer_rx_data = devm_kzalloc(ts->dev, ts->spi_xfer_max_sz, GFP_KERNEL); + if (!ts->xfer_rx_data) + return -ENOMEM; + + ts->xfer_tx_data = devm_kzalloc(ts->dev, ts->spi_xfer_max_sz, GFP_KERNEL); + if (!ts->xfer_tx_data) + return -ENOMEM; + + spin_lock_init(&ts->irq_lock); + mutex_init(&ts->rw_lock); + mutex_init(&ts->reg_lock); + dev_set_drvdata(&spi->dev, ts); + spi_set_drvdata(spi, ts); + + ts->probe_finish = false; + ts->initialized = false; + ts->ic_boot_done = false; + + ret = himax_platform_init(ts); + if (ret) { + dev_err(ts->dev, "%s: platform init failed\n", __func__); + return ret; + } + + ret = hx83102j_chip_detect(ts); + if (ret) { + dev_err(ts->dev, "%s: IC detect failed\n", __func__); + return ret; + } + + ret = himax_chip_init(ts); + if (ret < 0) + return ret; + ts->probe_finish = true; + + return ret; + himax_platform_deinit(ts); +} + +/** + * himax_spi_drv_remove - Remove function for the SPI driver + * @spi: Pointer to the spi_device structure + * + * This function is called when the SPI driver is removed. It deinitializes the + * himax_ts_data structure and free the resources allocated for the SPI + * communication. + */ +static void himax_spi_drv_remove(struct spi_device *spi) +{ + struct himax_ts_data *ts = spi_get_drvdata(spi); + + if (ts->probe_finish) { + if (ts->ic_boot_done) { + himax_int_enable(ts, false); + + if (ts->hid_probed) + himax_hid_remove(ts); + } + himax_platform_deinit(ts); + } +} + +/** + * himax_shutdown - Shutdown the touch screen + * @spi: Himax touch screen spi device + * + * This function is used to shutdown the touch screen. It will disable the + * interrupt, set the reset pin to activate state. Then remove the hid device. + * + * Return: None + */ +static void himax_shutdown(struct spi_device *spi) +{ + struct himax_ts_data *ts = spi_get_drvdata(spi); + + if (!ts->initialized) { + dev_err(ts->dev, "%s: init not ready, skip!\n", __func__); + return; + } + + himax_int_enable(ts, false); + gpiod_set_value(ts->pdata.gpiod_rst, 1); + himax_hid_remove(ts); +} + +#if defined(CONFIG_OF) +static const struct of_device_id himax_table[] = { + { .compatible = "himax,hx83102j" }, + {}, +}; +MODULE_DEVICE_TABLE(of, himax_table); +#endif + +static struct spi_driver himax_hid_over_spi_driver = { + .driver = { + .name = "hx83102j", + .owner = THIS_MODULE, +#if defined(CONFIG_OF) + .of_match_table = of_match_ptr(himax_table), +#endif + }, + .probe = himax_spi_drv_probe, + .remove = himax_spi_drv_remove, + .shutdown = himax_shutdown, +}; + +static int __init himax_ic_init(void) +{ + return spi_register_driver(&himax_hid_over_spi_driver); +} + +static void __exit himax_ic_exit(void) +{ + spi_unregister_driver(&himax_hid_over_spi_driver); +} + +module_init(himax_ic_init); +module_exit(himax_ic_exit); + +MODULE_VERSION("1.3.4"); +MODULE_DESCRIPTION("Himax HX83102J SPI driver for HID"); +MODULE_AUTHOR("Himax, Inc."); +MODULE_LICENSE("GPL"); diff --git a/drivers/hid/hid-himax-83102j.h b/drivers/hid/hid-himax-83102j.h new file mode 100644 index 000000000000..1b8e8904e9ab --- /dev/null +++ b/drivers/hid/hid-himax-83102j.h @@ -0,0 +1,286 @@ +/* SPDX-License-Identifier: GPL-2.0 */ +/* + * Himax hx83102j SPI Driver Code for HID. + * + * Copyright (C) 2024 Himax Corporation. + */ + +#ifndef __HID_HIMAX_83102J_H__ +#define __HID_HIMAX_83102J_H__ +// #define HX_PWR_CONFIG + +#include +#include +#include +#include +#include +#include + +/* Constants */ +#define HIMAX_BUS_RETRY 3 +/* SPI bus read header length */ +#define HIMAX_BUS_R_HLEN 3U +/* SPI bus write header length */ +#define HIMAX_BUS_W_HLEN 2U +/* TP SRAM address size and data size */ +#define HIMAX_REG_SZ 4U +/* TP MAX TX/RX number */ +#define HIMAX_MAX_RX 144U +#define HIMAX_MAX_TX 48U +/* TP FW main code size */ +#define HIMAX_FW_MAIN_CODE_SZ 0x20000U +/* TP max event data size */ +#define HIMAX_MAX_TP_EVENT_SZ 128U +/* TP max interrupt data size : event size + raw data size(rx * tx 16bits) */ +#define HIMAX_MAX_TP_EV_STACK_SZ (HIMAX_BUS_R_HLEN + \ + HIMAX_MAX_TP_EVENT_SZ + \ + (HIMAX_MAX_RX * HIMAX_MAX_TX) * 2) +/* SPI Max read/write size for max possible usage, write FW main code */ +#define HIMAX_BUS_RW_MAX_LEN (HIMAX_FW_MAIN_CODE_SZ + \ + HIMAX_BUS_W_HLEN + HIMAX_REG_SZ) +/* SPI CS setup time */ +#define HIMAX_SPI_CS_SETUP_TIME 300 +/* HIDRAW report header size */ +#define HIMAX_HID_REPORT_HDR_SZ 2U +/* hx83102j IC parameters */ +#define HIMAX_HX83102J_DSRAM_SZ 73728U +#define HIMAX_HX83102J_FLASH_SIZE 261120U +#define HIMAX_HX83102J_MAX_RX_NUM 50U +#define HIMAX_HX83102J_MAX_TX_NUM 32U +#define HIMAX_HX83102J_PAGE_SIZE 128U +#define HIMAX_HX83102J_SAFE_MODE_PASSWORD 0x9527 +#define HIMAX_HX83102J_STACK_SIZE 128U +#define HIMAX_HX83102J_FULL_STACK_SZ \ + (HIMAX_HX83102J_STACK_SIZE + \ + (2 + HIMAX_HX83102J_MAX_RX_NUM * HIMAX_HX83102J_MAX_TX_NUM + \ + HIMAX_HX83102J_MAX_TX_NUM + HIMAX_HX83102J_MAX_RX_NUM) * 2) +#define HIMAX_HX83102J_REG_XFER_MAX 4096U +/* AHB register addresses */ +#define HIMAX_AHB_ADDR_BYTE_0 0x00 +#define HIMAX_AHB_ADDR_RDATA_BYTE_0 0x08 +#define HIMAX_AHB_ADDR_ACCESS_DIRECTION 0x0c +#define HIMAX_AHB_ADDR_INCR4 0x0d +#define HIMAX_AHB_ADDR_CONTI 0x13 +#define HIMAX_AHB_ADDR_EVENT_STACK 0x30 +#define HIMAX_AHB_ADDR_PSW_LB 0x31 +/* AHB register values/commands */ +#define HIMAX_AHB_CMD_ACCESS_DIRECTION_READ 0x00 +#define HIMAX_AHB_CMD_CONTI 0x31 +#define HIMAX_AHB_CMD_INCR4 0x10 +#define HIMAX_AHB_CMD_INCR4_ADD_4_BYTE 0x01 +#define HIMAX_AHB_CMD_LEAVE_SAFE_MODE 0x0000 +/* hx83102j-specific register/dsram flags/data */ +#define HIMAX_HX83102J_REG_ADDR_TCON_RST 0x80020004 +/* hardware register addresses */ +#define HIMAX_REG_ADDR_SPI200_DATA 0x8000002c +#define HIMAX_REG_ADDR_CTRL_FW 0x9000005c +#define HIMAX_REG_ADDR_FW_STATUS 0x900000a8 +#define HIMAX_REG_ADDR_ICID 0x900000d0 +/* hardware reg data/flags */ +#define HIMAX_REG_DATA_FW_STATE_RUNNING 0x05 +#define HIMAX_REG_DATA_FW_STATE_SAFE_MODE 0x0c +#define HIMAX_REG_DATA_FW_RE_INIT 0x00 +#define HIMAX_REG_DATA_FW_GO_SAFEMODE 0xa5 +#define HIMAX_REG_DATA_FW_IN_SAFEMODE 0x87 +#define HIMAX_REG_DATA_ICID 0x83102900 +#define HIMAX_REG_DATA_TCON_RST 0x00000000 +/* HIMAX SPI function select, 1st byte of any SPI command sequence */ +#define HIMAX_SPI_FUNCTION_READ 0xf3 +#define HIMAX_SPI_FUNCTION_WRITE 0xf2 +/* Map code of FW 1k header */ +#define HIMAX_TP_CONFIG_TABLE 0x00000a00 +#define HIMAX_FW_CID 0x10000000 +#define HIMAX_FW_VER 0x10000100 +#define HIMAX_CFG_VER 0x10000600 +#define HIMAX_HID_TABLE 0x30000100 +#define HIMAX_FW_BIN_DESC 0x10000000 + +/** + * enum himax_hidraw_id_function - HIDRAW report IDs + * @HIMAX_ID_CONTACT_COUNT: Contact count report ID + */ +enum himax_hidraw_id_function { + HIMAX_ID_CONTACT_COUNT = 0x03, +}; + +/** + * enum himax_touch_report_status - ts operation return code for touch report + * @HIMAX_TS_GET_DATA_FAIL: Get touch data fail + * @HIMAX_TS_SUCCESS: Get touch data success + */ +enum himax_touch_report_status { + HIMAX_TS_GET_DATA_FAIL = -4, + HIMAX_TS_SUCCESS = 0, +}; + +/** + * struct himax_fw_address_table - address/offset in firmware image + * @addr_fw_ver_major: Address to Major version of firmware + * @addr_fw_ver_minor: Address to Minor version of firmware + * @addr_cfg_ver_major: Address to Major version of config + * @addr_cfg_ver_minor: Address to Minor version of config + * @addr_cid_ver_major: Address to Major version of Customer ID + * @addr_cid_ver_minor: Address to Minor version of Customer ID + * @addr_cfg_table: Address to Configuration table + * @addr_hid_table: Address to HID tables start offset + * @addr_hid_desc: Address to HID descriptor table + * @addr_hid_rd_desc: Address to HID report descriptor table + */ +struct himax_fw_address_table { + u32 addr_fw_ver_major; + u32 addr_fw_ver_minor; + u32 addr_cfg_ver_major; + u32 addr_cfg_ver_minor; + u32 addr_cid_ver_major; + u32 addr_cid_ver_minor; + u32 addr_cfg_table; + u32 addr_hid_table; + u32 addr_hid_desc; + u32 addr_hid_rd_desc; +}; + +/** + * struct himax_hid_rd_data - HID report descriptor data + * @rd_data: Point to report description data + * @rd_length: Length of report description data + */ +struct himax_hid_rd_data { + u8 *rd_data; + u32 rd_length; +}; + +/** + * union himax_dword_data - 4 bytes data union + * @dword: 1 dword data + * @word: 2 words data in word array + * @byte: 4 bytes data in byte array + */ +union himax_dword_data { + u32 dword; + u16 word[2]; + u8 byte[4]; +}; + +/** + * struct himax_ic_data - IC information holder + * @stylus_ratio: Stylus ratio + * @rx_num: Number of RX + * @tx_num: Number of TX + * @button_num: Number of buttons + * @max_point: Maximum touch point + * @icid: IC ID + * @interrupt_is_edge: Interrupt is edge otherwise level + * @stylus_function: Stylus function available or not + * @stylus_v2: Is stylus version 2 + */ +struct himax_ic_data { + u8 stylus_ratio; + u32 rx_num; + u32 tx_num; + u32 button_num; + u32 max_point; + u32 icid; + bool interrupt_is_edge; + bool stylus_function; + bool stylus_v2; +}; + +/** + * struct himax_hid_desc - HID descriptor + * @desc_length: Length of HID descriptor + * @bcd_version: BCD version + * @report_desc_length: Length of report descriptor + * @max_input_length: Maximum input length + * @max_output_length: Maximum output length + * @max_fragment_length: Maximum fragment length + * @vendor_id: Vendor ID + * @product_id: Product ID + * @version_id: Version ID + * @flags: Flags + * @reserved: Reserved + * + * This structure is used to hold the HID descriptor. + * It directly maps to a sequence of bytes in firmware image, + * thus need to be packed. + */ +struct himax_hid_desc { + u16 desc_length; + u16 bcd_version; + u16 report_desc_length; + u16 max_input_length; + u16 max_output_length; + u16 max_fragment_length; + u16 vendor_id; + u16 product_id; + u16 version_id; + u16 flags; + u32 reserved; +} __packed; + +/** + * struct himax_platform_data - Platform data holder + * @gpiod_rst: GPIO reset + * + * This structure is used to hold the platform related data. + */ +struct himax_platform_data { + struct gpio_desc *gpiod_rst; +}; + +/** + * struct himax_ts_data - Touchscreen data holder + * @xfer_buf: Interrupt data buffer + * @xfer_rx_data: SPI Transfer receive data buffer + * @xfer_tx_data: SPI Transfer transmit data buffer + * @himax_fw_data: Firmware data holder from flash + * @himax_irq: IRQ number + * @spi_xfer_max_sz: Size of SPI controller max transfer size + * @xfer_buf_sz: Size of interrupt data buffer + * @irq_state: IRQ state + * @irq_lock: Spin lock for irq + * @initialized: Indicate the driver is initialized + * @probe_finish: Indicate the driver probe is finished + * @ic_boot_done: Indicate the IC boot is done + * @hid_probed: Indicate the HID device is probed + * @touch_data_sz: Size of each interrupt data from IC + * @dev: Device pointer + * @spi: SPI device pointer + * @hid: HID device pointer + * @reg_lock: Mutex lock for reg access + * @rw_lock: Mutex lock for read/write action + * @ic_data: IC information holder + * @pdata: Platform data holder + * @fw_info_table: Firmware information address table of firmware image + * @hid_desc: HID descriptor + * @hid_rd_data: HID report descriptor data + */ +struct himax_ts_data { + u8 *xfer_buf; + u8 *xfer_rx_data; + u8 *xfer_tx_data; + u8 *himax_fw_data; + s32 himax_irq; + u32 spi_xfer_max_sz; + u32 xfer_buf_sz; + atomic_t irq_state; + /* lock for irq_save */ + spinlock_t irq_lock; + bool initialized; + bool probe_finish; + bool ic_boot_done; + bool hid_probed; + int touch_data_sz; + struct device *dev; + struct spi_device *spi; + struct hid_device *hid; + /* lock for register operation */ + struct mutex reg_lock; + /* lock for bus read/write action */ + struct mutex rw_lock; + struct himax_ic_data ic_data; + struct himax_platform_data pdata; + struct himax_fw_address_table fw_info_table; + struct himax_hid_desc hid_desc; + struct himax_hid_rd_data hid_rd_data; +}; +#endif From patchwork Tue Apr 2 10:49:29 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Allen_Lin X-Patchwork-Id: 785571 Received: from APC01-PSA-obe.outbound.protection.outlook.com (mail-psaapc01olkn2017.outbound.protection.outlook.com [40.92.52.17]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 84FCC5FDDB; Tue, 2 Apr 2024 10:50:18 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=fail smtp.client-ip=40.92.52.17 ARC-Seal: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055020; cv=fail; b=dQACchtaVCEb3EgdAAq+Tlhe9FVetJah1kMW7wfOtQAdT+N1cHZUUyEbkCVVhsRy4joL1oznKdxgI/3hDJN23t1WNuBdryYUZ7l7G+OlqMbtxL76CxY+wXY3OOZYMHQlFTQG7itqr7auLl6iHxRCEbMW3QNueuKA8SlVao2N0VI= ARC-Message-Signature: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055020; c=relaxed/simple; bh=CE9rJ6YqiZIptrf0u8T9LOLW+ImUjKp8wbTlBvnPwUg=; h=From:To:Cc:Subject:Date:Message-ID:In-Reply-To:References: Content-Type:MIME-Version; b=YeVaILlhvfcj2lY4IBV3uRaj3zXAkDl2WM5r0d6bzrSIpy6HqngZHCpqdviXSS+CzPM2C3gbepTR5R6FWVPBXl7FloB59215yPUZ/aQsH0bwdCDokMgzMMVhhrPoKQj/COnTx61Uece7FTLPPcEor6JZYp/ovI6Va4hInAVXE6c= ARC-Authentication-Results: i=2; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com; spf=pass smtp.mailfrom=hotmail.com; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b=tULaxgsK; arc=fail smtp.client-ip=40.92.52.17 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=hotmail.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b="tULaxgsK" ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=O2JDDWQ+SRSQWWbzfCiBXlUNAW2qcgeT3b1z/uV77zd8uEqLjlf/sXB+9mCH7khYabjTEyIIHLwCz/3jau56wyYyxCsZBn//vURatwtdJ3wzljHHWSJKomiSWoSSKSvIXbFachQOmNd17VEa8wH1iEIeg0xLRzR8+lMeyIepMpTTEnrzUebkniCoIssS9+xe/g5KAQnALkfaLsea2JMbdYWLnJUmEdA03bdmLyXvFEdiJVD3eV9JXVD1cIE1EZrbPxvCi4Lnm7eH5mpjRiA0aiTk/a7/qHEFNBfNUg38yzl/h2BC39R9bbag3STcb/noRn6b5s+/HYb4CMyfc3jjxA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=pQJyGZPwRlVI7ReqM3YuMwvJE6Hi08CgUoAmyBY8io8=; b=WNUVCHiWbApgs7T43fMfiazTUvz30YZNPkqPpDGwaG0t18RpXkNcNQgnW6H2MQbmVwxI4m+Ml+l8m8fAFvyjFG82W9MsQUBIyNEghOLnvA8VrJ8TIrWo7zHQt3U4kuUijbNvvQzXR4nfg82v36k+yRkg25YJKOQttuwUvoc6sK+ujekfBQ2JB7BbDmW9LdvP8WgNHvTWZy64euYnrMAeYhEfTaQawTiJ5CdRYQXa+X1BcZ+qoWQC7HGUeie/k1JfzxUSg/YaZCK69spPvbwBzpPY1KhY5/CshFXNPmmmlpvrrxRuuaSmNoKL0cxcH+8IWAVar58RRRmjfr5gyuQ3Yg== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=none; dmarc=none; dkim=none; arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=hotmail.com; s=selector1; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=pQJyGZPwRlVI7ReqM3YuMwvJE6Hi08CgUoAmyBY8io8=; b=tULaxgsKmayDXFDY6RCVVN3ciDeFGOXY4LQuxFjzaV16mE/rSDez7RkmExmrK3SHWO5GZm5aJwH5hY71KjNZCuhVBP4vZ3Q6hXrZwmzCcJC/sCIRNW6zEmLDWzB7qwSjZITH722Suj+KCDgq2/Z9q5qJeVblFw8BJXTm4CoYaQHzqcu4xrX4ibzdmmY4j2cOpGWisyvwcQboBRm8gI6RgLNBVfZFaO2mA/9xoigOtTYXZWCsh+uS6yPWmcIIDeGndxrJXfrSFWlCdUDs4xtQqE14ENRgPk2L7ihI5lbCQcMxDT/F9ocIZnshD9MoFy3cpU3EOLR/utg/b9HXkTyMYA== Received: from TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) by KL1PR06MB6964.apcprd06.prod.outlook.com (2603:1096:820:121::11) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.7409.46; Tue, 2 Apr 2024 10:50:11 +0000 Received: from TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf]) by TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf%5]) with mapi id 15.20.7409.042; Tue, 2 Apr 2024 10:50:11 +0000 From: Allen_Lin To: dmitry.torokhov@gmail.com, robh@kernel.org, krzysztof.kozlowski+dt@linaro.org, conor@kernel.org, jikos@kernel.org, benjamin.tissoires@redhat.com, linux-input@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Cc: Allen_Lin Subject: [PATCH v2 3/4] HID: Add DRM panel follower function Date: Tue, 2 Apr 2024 18:49:29 +0800 Message-ID: X-Mailer: git-send-email 2.34.1 In-Reply-To: <20240402104930.1053016-1-allencl_lin@hotmail.com> References: <20240402104930.1053016-1-allencl_lin@hotmail.com> X-TMN: [k++IVqfsyqAuCcDwHd1UvUQqDgLH7aBG] X-ClientProxiedBy: PS2PR02CA0031.apcprd02.prod.outlook.com (2603:1096:300:59::19) To TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) X-Microsoft-Original-Message-ID: <20240402104930.1053016-4-allencl_lin@hotmail.com> Precedence: bulk X-Mailing-List: linux-input@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 X-MS-Exchange-MessageSentRepresentingType: 1 X-MS-PublicTrafficType: Email X-MS-TrafficTypeDiagnostic: TY0PR06MB5611:EE_|KL1PR06MB6964:EE_ X-MS-Office365-Filtering-Correlation-Id: 0647b671-9b50-47d2-6444-08dc5302ab69 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: gPrN+4I2tpGvJAwL7+je9xVEGLj7f/2CyQulriHLvNWPeD73r8SwGaI7sLho2jf7C2mhgkBOXQZPyy1urKcdcU3txwnHH9NO8uhCckadCJ8g6oCeBoSxHgNsZ9Mf9RSERPw4oJiSnBiPjvQl5/3xWicyIPup4/e132BOZ91uRNChsOXHWov6RedfMfL9tbkXMVuWKNpNaK19+s3/ogE7xrQYVF4636jVbmTrsNgIs7zvuboMjFLnlAWAQmpiEc1eqA0u+hLe1QYEb7dhv04fOnoFoK3cfX+4qKlX/xqal9DXngiy6wTDK67s6Yu+KZkqpZIh/n/wOecDytORS2s6zefbjMyRbTKCYi2tJpHJmOcnI9tj1l/6JiESPtkhnfxWjHWkCsR3C8pwNKlv6GeG6CM6hvcQeCXnowUpZ3p481G95Ta4eUjtaKZt5Vx7ZpE6MEv2Y7HF/eo4Fyv846WLKQ03D+aBXDyGZq2VeEArvojC88aDIv6YTxFKGL/CaSBHxIVoFa3ttwYXRJ+bf36f8xKOV+yQv1tLCz5h01fHwmhmf4/lAylh/A6W40Bm0AyA7VPe//aAMmOcnyrLiWIVRjbuCo8EESS5rH19TuMjpBI/idgSv7e17BqyW+FWQTGSAP/wKPgECQ2/yFfH6b/wnDug5sYZrZvDwNbwQDqP/NcmvLfJ72X3P4HToFeTjqYa X-MS-Exchange-AntiSpam-MessageData-ChunkCount: 1 X-MS-Exchange-AntiSpam-MessageData-0: Oysl8BxOn2rY0Dcm5/dZCQ0TAQ8cuvbA/qd0Voq8b4N7cZoTVAqDSZRWRjNE2Hs0sSxRmSZRz4dQdGKSjosH2TZVGrgES50nembvk4rxXIKOuk/Kd5jqHCBDPIHQJTbzg7nAFS08vKcGw3Jt3x9RPVrfIc8SxrYVp9qu7V2bz2H/ZdW0CrXgTqHi4pl3v/zwrTzDnGD2NCSxTyr6YiDqRpbKo5fuyZbXFzimrYHzzahAyPymXlPQfkq5Pyo+N4QVz/MPxraoAI0dg81654/vjDrxuY7PnLEwnzINctgdLYd/Qx5iuPft3HVReHkFVsIddGPDL6DyNKRXfLainzdwwMu2mI91PunfjRjajRkCsNKIVrr+Ew+GFhDjGaYUNCO+Ch94xDVZjjH/oW5RBA0OA6pu9Vq37+krL7el3RQP/DKRDOnQ02a2ZoRig/uP8QPN8syXuu9rUNynmKMOMSoR8PZMHyYHGiEvl2mihjgAdSYG2i5CRs/HQxbGJl3YFcnCToYa5CrnnS3C7O909Z72QUKhbGzorgYAq2W5CDKuUrnxNCuaU1DQoWAyVn7ihdUcKQTKynNopIwuQvWoqxC/so2qf52jiePHNIEX7X9ZXXHu0XhCD8M+NysWBWu4q4zotFCyJZZ2gCSgjWfI6qpF7laaB2hdakayGdSkF5EiJ53sAojEDfrdL9CpC3Gb9shCuthoLW4il9Lg6BFd2AhcSq5qy17p4J5DmOr9WGU30XJISo7jjEsov+1kso9IDZmUAKO8AZryY000TlEItRHp8Oi2E0GcO0bmEUfJvhTLAAq+UTU/OmZ+vp4J6q7RfG7hUVHCl4mWUuh924alCOPVmxSYiDYAW3dtJqq2iFwpKgMlNhNMyHt7KhvP5N7/oHOk4TXR9XxOJNx2P0MH9cS7OmNnqurSZtwsrOaQCqho5/nN0pUZtRc2OICGQybtQ/vnbEyOx4m1cHhILSbuJlSjtYStTA84JExDHJllzwCxfNm7alW/ijeiEHwXemZtPn13tlAjIEOSUAYaeywgFoFZFGLFVFs7C93QYg2WbS3yFtEDDTZAMC/YF6NrxZ6E13DeYLvdTJRQ2bjciWIXbIrLAIhAWfypk2VgBggeBH0Ew57Dfv2khfy+H5JRTpp36JJgfk+MbsHn4o33gkNLtE/gPEIGvVECdLALpn3unbBvnLRGzLjhEPPCYxYIBshxAN876N1X7q9gsg4PFKfUvKwyuEtd/Rmr0vmTa0JugWDfyBQ= X-OriginatorOrg: sct-15-20-4755-11-msonline-outlook-3208f.templateTenant X-MS-Exchange-CrossTenant-Network-Message-Id: 0647b671-9b50-47d2-6444-08dc5302ab69 X-MS-Exchange-CrossTenant-AuthSource: TY0PR06MB5611.apcprd06.prod.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Internal X-MS-Exchange-CrossTenant-OriginalArrivalTime: 02 Apr 2024 10:50:11.3923 (UTC) X-MS-Exchange-CrossTenant-FromEntityHeader: Hosted X-MS-Exchange-CrossTenant-Id: 84df9e7f-e9f6-40af-b435-aaaaaaaaaaaa X-MS-Exchange-CrossTenant-RMS-PersistedConsumerOrg: 00000000-0000-0000-0000-000000000000 X-MS-Exchange-Transport-CrossTenantHeadersStamped: KL1PR06MB6964 Add DRM panel follower[1] to trigger suspend/resume due to TDDI nature, TP need to follow panel power sequence. [1]: https://lore.kernel.org/all/20230727171750.633410-1-dianders@chromium.org Signed-off-by: Allen_Lin --- drivers/hid/hid-himax-83102j.c | 289 +++++++++++++++++++++++++++++++-- drivers/hid/hid-himax-83102j.h | 10 ++ 2 files changed, 283 insertions(+), 16 deletions(-) diff --git a/drivers/hid/hid-himax-83102j.c b/drivers/hid/hid-himax-83102j.c index aa8d0b6677bb..0a2be071a6c4 100644 --- a/drivers/hid/hid-himax-83102j.c +++ b/drivers/hid/hid-himax-83102j.c @@ -8,6 +8,7 @@ #include "hid-himax-83102j.h" static int himax_chip_init(struct himax_ts_data *ts); +static int himax_platform_init(struct himax_ts_data *ts); static void himax_ts_work(struct himax_ts_data *ts); /** @@ -1135,7 +1136,6 @@ static int himax_hid_probe(struct himax_ts_data *ts) hid = ts->hid; if (hid) { - dev_warn(ts->dev, "%s: hid device already exist!\n", __func__); hid_destroy_device(hid); hid = NULL; } @@ -1443,6 +1443,151 @@ static int himax_load_config(struct himax_ts_data *ts) return -EINVAL; } +/** + * himax_ap_notify_fw_suspend() - Notify the FW of AP suspend status + * @ts: Himax touch screen data + * @suspend: Suspend status, true for suspend, false for resume + * + * This function is used to notify the FW of AP suspend status. It will write + * the suspend status to the DSRAM and read the status back to check if the + * status is written successfully. If IC is powered off when suspend, this + * function will only be used when resume. + * + * Return: None + */ +static void himax_ap_notify_fw_suspend(struct himax_ts_data *ts, bool suspend) +{ + int ret; + u32 retry_cnt; + const u32 retry_limit = 10; + union himax_dword_data rdata, data; + + if (suspend) + data.dword = cpu_to_le32(HIMAX_DSRAM_DATA_AP_NOTIFY_FW_SUSPEND); + else + data.dword = cpu_to_le32(HIMAX_DSRAM_DATA_AP_NOTIFY_FW_RESUME); + + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + ret = himax_mcu_register_write(ts, HIMAX_DSRAM_ADDR_AP_NOTIFY_FW_SUSPEND, + data.byte, 4); + if (ret) { + dev_err(ts->dev, "%s: write suspend status failed!\n", __func__); + return; + } + usleep_range(1000, 1100); + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_AP_NOTIFY_FW_SUSPEND, + rdata.byte, 4); + if (ret) { + dev_err(ts->dev, "%s: read suspend status failed!\n", __func__); + return; + } + + if (rdata.dword == data.dword) + break; + } +} + +/** + * himax_resume_proc() - Chip resume procedure of touch screen + * @ts: Himax touch screen data + * + * This function is used to resume the touch screen. It will call the + * himax_ap_notify_fw_suspend() to notify the FW of AP resume status. + * + * Return: None + */ +static void himax_resume_proc(struct himax_ts_data *ts) +{ + himax_ap_notify_fw_suspend(ts, false); +} + +/** + * himax_chip_suspend() - Suspend the touch screen + * @ts: Himax touch screen data + * + * This function is used to suspend the touch screen. It will disable the + * interrupt and set the reset pin to activate state. Remove the HID at + * the end, to prevent stuck finger when resume. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_chip_suspend(struct himax_ts_data *ts) +{ + himax_int_enable(ts, false); + gpiod_set_value(ts->pdata.gpiod_rst, 1); + himax_hid_remove(ts); + + return 0; +} + +/** + * himax_chip_resume() - Setup flags, I/O and resume + * @ts: Himax touch screen data + * + * This function is used to resume the touch screen. It will set the resume + * success flag to false, and disable reset pin. Then call the himax_resume_proc() + * to process detailed resume procedure. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_chip_resume(struct himax_ts_data *ts) +{ + gpiod_set_value(ts->pdata.gpiod_rst, 0); + himax_resume_proc(ts); + himax_hid_probe(ts); + himax_int_enable(ts, true); + + return 0; +} + +/** + * himax_suspend() - Suspend the touch screen + * @dev: Device structure + * + * Wrapper function for himax_chip_suspend() to be called by the PM or + * the DRM panel notifier. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_suspend(struct device *dev) +{ + struct himax_ts_data *ts = dev_get_drvdata(dev); + + if (!ts->initialized) { + dev_err(ts->dev, "%s: init not ready, skip!\n", __func__); + return -ECANCELED; + } + himax_chip_suspend(ts); + + return 0; +} + +/** + * himax_resume() - Resume the touch screen + * @dev: Device structure + * + * Wrapper function for himax_chip_resume() to be called by the PM or + * the DRM panel notifier. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_resume(struct device *dev) +{ + int ret; + struct himax_ts_data *ts = dev_get_drvdata(dev); + + if (!ts->initialized) { + if (himax_chip_init(ts)) + return -ECANCELED; + } + + ret = himax_chip_resume(ts); + if (ret < 0) + dev_err(ts->dev, "%s: resume failed!\n", __func__); + + return ret; +} + /** * himax_chip_init() - Initialize the Himax touch screen * @ts: Himax touch screen data @@ -1473,6 +1618,130 @@ static int himax_chip_init(struct himax_ts_data *ts) return 0; } +/** + * __himax_initial_power_up() - Initial power up of the Himax touch screen + * @ts: Himax touch screen data + * + * This function is used to perform the initial power up sequence of the Himax + * touch screen for DRM panel notifier. + * + * Return: 0 on success, negative error code on failure + */ +static int __himax_initial_power_up(struct himax_ts_data *ts) +{ + int ret; + + ret = himax_platform_init(ts); + if (ret) { + dev_err(ts->dev, "%s: platform init failed\n", __func__); + return ret; + } + + ret = hx83102j_chip_detect(ts); + if (ret) { + dev_err(ts->dev, "%s: IC detect failed\n", __func__); + return ret; + } + + ret = himax_chip_init(ts); + if (ret) { + dev_err(ts->dev, "%s: chip init failed\n", __func__); + return ret; + } + ts->probe_finish = true; + + return 0; +} + +/** + * himax_panel_prepared() - Panel prepared callback + * @follower: DRM panel follower + * + * This function is called when the panel is prepared. It will call the + * __himax_initial_power_up() when the probe is not finished which means + * the first time driver start. Otherwise, it will call the himax_resume() + * to performed resume process. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_panel_prepared(struct drm_panel_follower *follower) +{ + struct himax_platform_data *pdata = + container_of(follower, struct himax_platform_data, panel_follower); + struct himax_ts_data *ts = container_of(pdata, struct himax_ts_data, pdata); + + if (!ts->probe_finish) + return __himax_initial_power_up(ts); + else + return himax_resume(ts->dev); +} + +/** + * himax_panel_unpreparing() - Panel unpreparing callback + * @follower: DRM panel follower + * + * This function is called when the panel is unpreparing. It will call the + * himax_suspend() to perform the suspend process. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_panel_unpreparing(struct drm_panel_follower *follower) +{ + struct himax_platform_data *pdata = + container_of(follower, struct himax_platform_data, panel_follower); + struct himax_ts_data *ts = container_of(pdata, struct himax_ts_data, pdata); + + return himax_suspend(ts->dev); +} + +/* Panel follower function table */ +static const struct drm_panel_follower_funcs himax_panel_follower_funcs = { + .panel_prepared = himax_panel_prepared, + .panel_unpreparing = himax_panel_unpreparing, +}; + +/** + * himax_register_panel_follower() - Register the panel follower + * @ts: Himax touch screen data + * + * This function is used to register the panel follower. It will set the + * pdata.is_panel_follower to true and register the panel follower. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_register_panel_follower(struct himax_ts_data *ts) +{ + struct device *dev = ts->dev; + + ts->pdata.is_panel_follower = true; + ts->pdata.panel_follower.funcs = &himax_panel_follower_funcs; + + if (device_can_wakeup(dev)) { + dev_warn(ts->dev, "Can't wakeup if following panel"); + device_set_wakeup_capable(dev, false); + } + + return drm_panel_add_follower(dev, &ts->pdata.panel_follower); +} + +/** + * himax_initial_power_up() - Initial power up of the Himax touch screen + * @ts: Himax touch screen data + * + * This function checks if the device is a panel follower and calls + * himax_register_panel_follower() if it is. Otherwise, it calls + * __himax_initial_power_up(). + * + * Return: 0 on success, negative error code on failure + */ +static int himax_initial_power_up(struct himax_ts_data *ts) +{ + if (drm_is_panel_follower(ts->dev)) + return himax_register_panel_follower(ts); + else + return __himax_initial_power_up(ts); +} + /** * himax_platform_deinit() - Deinitialize the platform related settings * @ts: Pointer to the himax_ts_data structure @@ -1588,25 +1857,13 @@ static int himax_spi_drv_probe(struct spi_device *spi) ts->initialized = false; ts->ic_boot_done = false; - ret = himax_platform_init(ts); - if (ret) { - dev_err(ts->dev, "%s: platform init failed\n", __func__); - return ret; - } - - ret = hx83102j_chip_detect(ts); + ret = himax_initial_power_up(ts); if (ret) { - dev_err(ts->dev, "%s: IC detect failed\n", __func__); - return ret; + dev_err(ts->dev, "%s: initial power up failed\n", __func__); + return -ENODEV; } - ret = himax_chip_init(ts); - if (ret < 0) - return ret; - ts->probe_finish = true; - return ret; - himax_platform_deinit(ts); } /** diff --git a/drivers/hid/hid-himax-83102j.h b/drivers/hid/hid-himax-83102j.h index 1b8e8904e9ab..eef55c45b1d4 100644 --- a/drivers/hid/hid-himax-83102j.h +++ b/drivers/hid/hid-himax-83102j.h @@ -9,6 +9,7 @@ #define __HID_HIMAX_83102J_H__ // #define HX_PWR_CONFIG +#include #include #include #include @@ -69,6 +70,11 @@ #define HIMAX_AHB_CMD_INCR4 0x10 #define HIMAX_AHB_CMD_INCR4_ADD_4_BYTE 0x01 #define HIMAX_AHB_CMD_LEAVE_SAFE_MODE 0x0000 +/* DSRAM flag addresses */ +#define HIMAX_DSRAM_ADDR_AP_NOTIFY_FW_SUSPEND 0x10007fd0 +/* dsram flag data */ +#define HIMAX_DSRAM_DATA_AP_NOTIFY_FW_SUSPEND 0xa55aa55a +#define HIMAX_DSRAM_DATA_AP_NOTIFY_FW_RESUME 0x00000000 /* hx83102j-specific register/dsram flags/data */ #define HIMAX_HX83102J_REG_ADDR_TCON_RST 0x80020004 /* hardware register addresses */ @@ -219,11 +225,15 @@ struct himax_hid_desc { /** * struct himax_platform_data - Platform data holder + * @is_panel_follower: Is panel follower enabled + * @panel_follower: DRM panel follower * @gpiod_rst: GPIO reset * * This structure is used to hold the platform related data. */ struct himax_platform_data { + bool is_panel_follower; + struct drm_panel_follower panel_follower; struct gpio_desc *gpiod_rst; }; From patchwork Tue Apr 2 10:49:30 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Allen_Lin X-Patchwork-Id: 785232 Received: from APC01-PSA-obe.outbound.protection.outlook.com (mail-psaapc01olkn2017.outbound.protection.outlook.com [40.92.52.17]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 008EA69979; Tue, 2 Apr 2024 10:50:20 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=fail smtp.client-ip=40.92.52.17 ARC-Seal: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055024; cv=fail; b=e5TZPDPbrPB/eQKI3525LPXJNXwmW0sEE23vxNpguRjzpYh7k2Sb4FSqcCdDZFVUtjh+975KQf/6PgtJEMfiauLqMa+CMxZmanVJg1+MkjaddMDyD+Q1QUdFwPJiSFXDbIJtHwx8DhPOAPioZghT8e8JupZnKbepEpgGiIzzQtA= ARC-Message-Signature: i=2; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1712055024; c=relaxed/simple; bh=edDCafLcLl65gFMDXKfX5S4Fja1pk5kgNIJI7XXQPTc=; h=From:To:Cc:Subject:Date:Message-ID:In-Reply-To:References: Content-Type:MIME-Version; b=tnDjGcLwlODb//R9cgSRiQKM60Me8VfJzmROR4yow0S3AXHglEegDtUuAiGdspkszZQvoJBd9TzYtmHSmzDYRPZ5CCB7GYeowY+EH7jr27xEyc1tAzpBcs96ZCfj7LLgaQ3Q1x559Nn8+NrhMdALYPiaYg2QT/+Uo7MqUmW7ZoQ= ARC-Authentication-Results: i=2; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com; spf=pass smtp.mailfrom=hotmail.com; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b=alODJ22B; arc=fail smtp.client-ip=40.92.52.17 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=hotmail.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=hotmail.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=hotmail.com header.i=@hotmail.com header.b="alODJ22B" ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=OZzYbzpSnQXbLxzoaPjm3xIVxfnKZ5JYal5Qsr6qFGt9Z8PDQY8goPUhYVdSfUMAxolg1WuK/5B+flUpS03um4BwaiRh/9wDEBMJL+20pL//jKJ/0k3CMT6xqZXQdp3GxC/5ks0X4xhLApm8J2wrF426aCGZYvt4L7t93cdEXCcn9ISZu4+C8n5KP0BbivqhR4f8BMSfrnRU7V2nTxe/QlOiB3da8+JN2rTZPmscEC51MWPyar1deC5mulNIO2Rc3YYwouxs0FmpMfi1FXe46WKJ4WTZEKlfosCD9x3JxF3yuRVdZRVRNj1LbVFWBfR9MDg1GUaw/kIIgF97ZM+XHQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-AntiSpam-MessageData-ChunkCount:X-MS-Exchange-AntiSpam-MessageData-0:X-MS-Exchange-AntiSpam-MessageData-1; bh=6c0sZXMDtobf0JkosPqUTB1hsSwQ3O4MjGmtVBvOKZQ=; b=ISSeVP83kTVn//AgXg6e96fNqFLgGU9T9wtjfJep8N0XxSNGJ415rRwvcIIIdXlSn++TlhAjNM/l2LaYg3wO8gR1wSUSlgg8f4NlUShuoK3LUOExlB5fbRgWTHTHtuKhZ8mpxde2W43N76EG+EhLRykqab1cqPIoIEc+5lkpLRTllu8fmU/zd92lg0sp1FhjuwwomhGCNVxCYrfhfs/4qFa3zpsD7VDYL+uFaUpznVW8Ci17mgrxQH+7V8cCdHeK6yFYEQ37qQ3O67fSFL3II1DWW7U1KYEctoV67ycbTOTydZCJf1m2ff2/cKoGrbre+dJzGO5grmkD1FoNXkiR+g== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=none; dmarc=none; dkim=none; arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=hotmail.com; s=selector1; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=6c0sZXMDtobf0JkosPqUTB1hsSwQ3O4MjGmtVBvOKZQ=; b=alODJ22BPEpvQl2lGsDM5PhN+LX5lpJiQWW3LhN9WYfeo0G+ofPmyZcKkWuB1cy0p9t1JaM9fxyensFiy6gAF6l1RzQ4AUTkhG0m5Yn7Tv/5RljsB+6mBJAmTxfxcwMESu5jtGZD0UiSHhU+l6WrP78vpMifCTRy48dSvzzPXWGm5/lPNygyAfBEhV684uRYtC2DfPzbSot3HMqbkGk3e6YxL5uZPBS3JL4cmuOBHWU4xxKktLT5Tu7YnNuy43tv3mDb2k2oWW1dLOYXSOniKNDuw55iNqua0DDSuK2i/34txWuoTsGSUwuRjJhHYWC2pCxSe8PWoziHgbWVXiX5Cg== Received: from TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) by KL1PR06MB6964.apcprd06.prod.outlook.com (2603:1096:820:121::11) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.7409.46; Tue, 2 Apr 2024 10:50:15 +0000 Received: from TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf]) by TY0PR06MB5611.apcprd06.prod.outlook.com ([fe80::d57a:9f0e:1ee7:85bf%5]) with mapi id 15.20.7409.042; Tue, 2 Apr 2024 10:50:15 +0000 From: Allen_Lin To: dmitry.torokhov@gmail.com, robh@kernel.org, krzysztof.kozlowski+dt@linaro.org, conor@kernel.org, jikos@kernel.org, benjamin.tissoires@redhat.com, linux-input@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Cc: Allen_Lin Subject: [PATCH v2 4/4] HID: Load firmware directly from file to IC Date: Tue, 2 Apr 2024 18:49:30 +0800 Message-ID: X-Mailer: git-send-email 2.34.1 In-Reply-To: <20240402104930.1053016-1-allencl_lin@hotmail.com> References: <20240402104930.1053016-1-allencl_lin@hotmail.com> X-TMN: [4GlzUCwOXlL3duKHqyBVCfwNNwltSbBO] X-ClientProxiedBy: PS2PR02CA0031.apcprd02.prod.outlook.com (2603:1096:300:59::19) To TY0PR06MB5611.apcprd06.prod.outlook.com (2603:1096:400:31e::8) X-Microsoft-Original-Message-ID: <20240402104930.1053016-5-allencl_lin@hotmail.com> Precedence: bulk X-Mailing-List: linux-input@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 X-MS-Exchange-MessageSentRepresentingType: 1 X-MS-PublicTrafficType: Email X-MS-TrafficTypeDiagnostic: TY0PR06MB5611:EE_|KL1PR06MB6964:EE_ X-MS-Office365-Filtering-Correlation-Id: 2af4630c-0097-4fa4-c84e-08dc5302ada7 X-Microsoft-Antispam: BCL:0; X-Microsoft-Antispam-Message-Info: r/wMSXA+PV9PClI5hYLyH1LvuTfJHa2Np5DWjBYZ7SIFrBa15hDpQWQSW0lAZkPksuqkUQjZr2C5vyMPUk1kzexApVOYXVIIDws1sEqkodDnEnxWo61ynYYLzefqLmDgFBS14HQYliqmLkxj7bumo5GdB/IWf6wD1XGpTo69Aa4L3CfQyO2044Xn8ZwzV9xNpo/nwRAADZ9jcixuye/6IOG1MBji2/JEmDHSx/f/lBhUBRbIE4pGwloPnz8L2TRK4kJxmdv2b9Zt8vuA64c4mLbVpkaoN05/bhTgkWUGKN9cmK4NyeFhBx6iDqeD6dSguTrdMJy8BvcKDtnHaWWqQ+JxAhQsg04ZNUw9D+sizXLNX+JFCId3TgsFzylwxOXQXbzq/AtbI0MnK8oskvHd+jxjp6nG+W6Ysrt8uUrap41F2jtoFoVd/tQ2DvJwgwL7n26N0vGf9eaIYyNKM+BoeYUNlViINMimjfKHg630RvUkrb1AtMR8AeizP5GtvAtO5NQlVicSoqDxYNQKCZGCfbdRY2ZF51+rB8tUgytJpMuKcZuhwCHk6FKvh41JKn8yXhNf4s+dpxU6eel0TTktKtBun8n7qTkr0u8ROKBpZBk6JdvxXL2WKOba22hwgRLY X-MS-Exchange-AntiSpam-MessageData-ChunkCount: 1 X-MS-Exchange-AntiSpam-MessageData-0: xyGMwEw1uMLTaR6bDBFLHkre6TUJ0SgqvEebbTQ3XDVgtoV9jGzIHQTRZg+xVtYZM9U+bEP1/5uivHs8zmxoMw5zBVeot+mR5MhDYuP27l0a4K2cXHRiUaEESWk6tTouX2HywLiOlfkwnlG0gqCPXj7IO/9KTLAGGP1j52p/WRE1BcVsUHuC9X5W9eGfMljfyq0iVbMi94A7HtBiADG0sYPxWrtetx7VeAxTmd4NgPpSnpE2t46aych3H7bnjzxdIkxcCrZS/F3D1Y39WwDJu2DCAk/WgEHfM+91xP+gH9dBzBZqXQs1KYZBRHoApvNjP5bTqjh9bJrSwHZV+SyViom0svBwchuAiz4xfygF44qflUA6BVGg2NFE0zSwOsX/ouWVxGNkaIZ0bOBUeZWElUZ2abQDuU5iICoLP9gyPbQ3niV/IHufrv9jtkjfXIAbpPqccCydk4RKWqOWzv6LEkix1UuuaEvbmxQ00rFpVK9UgGSOeKOlp39LytsBBlQeGYNMd69+meEG16uohycAMhNEVaWRUw3TLFkmjQVDtT5bhyonKHJpdgfzKXOdKNY5PiOdOe2lgKQbR9v4ykH89QcMmixSiUfjY91yrH8m6P5GIgrsLzWjaDtjobgc0rpJgvSgPey/w33d5NnBUN3hjZ3V/6gKOUK8wGL7S+F5irHmQmccPUolBh9H4m8LLi6G9+li8whrCaXq8G5I3WlNsdgpYZuA28FQoyeX/gaNN152jv+oStVbCgietbN+23uvXZxgv0I4YlXpYHLf5boDUFIOlwoZCl5Um4eGoV9rCLwQfMReGcCYGUoGP+YrtxPCBeTu/kWLIiuXqxRstKGHRkhAHd0TIdFdMGhuymeoBuwLXR524p3AMRCVWYeBHqO8EBQe4CxYTiXFwzmrtJ/Oa2lV2e6tgpsKRRKJ/eHuZUxMesBNCfIHq0WUZQxc9HudiZ+ZVpzBeRZW49EcxMUUJoEjXtsPvtsJ3lP2ZpeHllZdqyOpkYstHRrYdNS4s+yTdGDITZuWF6N2Zbk3b33oOpkgHKzoJMa27KBEck8rlVq6BwdiZmheNKAuhDcBMSKRIAw3c3qOPSCd8sHwSYxJcdg9dYfFPz/qpSe8+kE0MBf6w5/mKef/m2osYk5YymXJhyjoLM7WttMwekK8sUoS8hgDEkA0teta0GXY5MAGyT+zHTAdihDa8zyNnTSJmuHs8SVWnSVnZxJiix1ONAb+szSbpomXMt0opGPT90jyQvo= X-OriginatorOrg: sct-15-20-4755-11-msonline-outlook-3208f.templateTenant X-MS-Exchange-CrossTenant-Network-Message-Id: 2af4630c-0097-4fa4-c84e-08dc5302ada7 X-MS-Exchange-CrossTenant-AuthSource: TY0PR06MB5611.apcprd06.prod.outlook.com X-MS-Exchange-CrossTenant-AuthAs: Internal X-MS-Exchange-CrossTenant-OriginalArrivalTime: 02 Apr 2024 10:50:15.3550 (UTC) X-MS-Exchange-CrossTenant-FromEntityHeader: Hosted X-MS-Exchange-CrossTenant-Id: 84df9e7f-e9f6-40af-b435-aaaaaaaaaaaa X-MS-Exchange-CrossTenant-RMS-PersistedConsumerOrg: 00000000-0000-0000-0000-000000000000 X-MS-Exchange-Transport-CrossTenantHeadersStamped: KL1PR06MB6964 For HX83102J, often shipped without flash for costdown purpose. This patch implement the function to load firmware file from user-space, write into IC sram and make it running. The procedure as following: 1. Check OF to get FW PID specified: himax_parse_dt 2. Load PID specified FW: i_get_FW, upload process use workqueue himax_boot_upgrade 3. Write FW into IC sram: i_update_FW 3-1. FW file contain a mapping table to indicate various section, such code, data, version info and HID info 3-2. Call update process himax_mcu_firmware_update_0f, which stop the internal MCU: i. system reset. ii. hx83102j_sense_off 3-3. Parsing FW and upload file: himax_zf_part_info. info[0] contain the code section and will upload to ISRAM. Other info[x] are data sections will be combined and upload to DSRAM separately. After upload FW, driver will use crc to check integrity: himax_sram_write_crc_check for code(HW CRC result must be 0), himax_mcu_calculate_crc_with_ap for pre-calculate data CRC[1], Upload data section by himax_mcu_register_write, himax_mcu_check_crc use HW calculate CRC and compared with [1] 4. Disable HW reload FW from flash function before MCU start running: himax_disable_fw_reload 5. Start running the FW: himax_mcu_power_on_init 5-1. Setting varies FW settings before FW start running 5-2. Start FW: hx83102j_sense_on 5-3. Wait until FW initial process complete: Read 0x72C0 from IC 6. Read HID info from FW 7. Read TP properties from IC: himax_mcu_tp_info_check 8. Read FW information fromIC: himax_mcu_read_FW_ver 9. Release FW, IC initial complete The resume procedure do part of boot process, it also need to reload FW from user-space into IC sram: himax_0f_op_file_dirly 1. Request FW 2. Upload FW: himax_mcu_firmware_update_0f 3. Disable HW reload FW from flash function: himax_disable_fw_reload 4. Start running the FW: himax_mcu_power_on_init Signed-off-by: Allen_Lin --- drivers/hid/hid-himax-83102j.c | 1515 +++++++++++++++++++++++++++----- drivers/hid/hid-himax-83102j.h | 168 +++- 2 files changed, 1483 insertions(+), 200 deletions(-) diff --git a/drivers/hid/hid-himax-83102j.c b/drivers/hid/hid-himax-83102j.c index 0a2be071a6c4..5cae979a3052 100644 --- a/drivers/hid/hid-himax-83102j.c +++ b/drivers/hid/hid-himax-83102j.c @@ -532,6 +532,92 @@ static void himax_mcu_ic_reset(struct himax_ts_data *ts, bool int_off) himax_int_enable(ts, true); } +/** + * hx83102j_reload_to_active() - Reload to active mode + * @ts: Himax touch screen data + * + * This function is used to write a flag to the IC register to make MCU restart without + * reload the firmware. + * + * Return: 0 on success, negative error code on failure + */ +static int hx83102j_reload_to_active(struct himax_ts_data *ts) +{ + int ret; + u32 retry_cnt; + const u32 addr = HIMAX_REG_ADDR_RELOAD_TO_ACTIVE; + const u32 reload_to_active_cmd = 0xec; + const u32 reload_to_active_done = 0x01ec; + const u32 retry_limit = 5; + union himax_dword_data data; + + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + data.dword = cpu_to_le32(reload_to_active_cmd); + ret = himax_mcu_register_write(ts, addr, data.byte, 4); + if (ret < 0) + return ret; + usleep_range(1000, 1100); + ret = himax_mcu_register_read(ts, addr, data.byte, 4); + if (ret < 0) + return ret; + data.dword = le32_to_cpu(data.dword); + if (data.word[0] == reload_to_active_done) + break; + } + + if (data.word[0] != reload_to_active_done) { + dev_err(ts->dev, "%s: Reload to active failed!\n", __func__); + return -EINVAL; + } + + return 0; +} + +/** + * hx83102j_en_hw_crc() - Enable/Disable HW CRC + * @ts: Himax touch screen data + * @en: true for enable, false for disable + * + * This function is used to enable or disable the HW CRC. The HW CRC + * is used to protect the SRAM data. + * + * Return: 0 on success, negative error code on failure + */ +static int hx83102j_en_hw_crc(struct himax_ts_data *ts, bool en) +{ + int ret; + u32 retry_cnt; + const u32 addr = HIMAX_HX83102J_REG_ADDR_HW_CRC; + const u32 retry_limit = 5; + union himax_dword_data data, wrt_data; + + if (en) + data.dword = cpu_to_le32(HIMAX_HX83102J_REG_DATA_HW_CRC); + else + data.dword = cpu_to_le32(HIMAX_HX83102J_REG_DATA_HW_CRC_DISABLE); + + wrt_data.dword = data.dword; + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + ret = himax_mcu_register_write(ts, addr, data.byte, 4); + if (ret < 0) + return ret; + usleep_range(1000, 1100); + ret = himax_mcu_register_read(ts, addr, data.byte, 4); + if (ret < 0) + return ret; + + if (data.word[0] == wrt_data.word[0]) + break; + } + + if (data.word[0] != wrt_data.word[0]) { + dev_err(ts->dev, "%s: ECC fail!\n", __func__); + return -EINVAL; + } + + return 0; +} + /** * hx83102j_sense_off() - Stop MCU and enter safe mode * @ts: Himax touch screen data @@ -641,6 +727,56 @@ static int hx83102j_sense_off(struct himax_ts_data *ts, bool check_en) return -EIO; } +/** + * hx83102j_sense_on() - Sense on the touch chip + * @ts: Himax touch screen data + * @sw_reset: true for software reset, false for hardware reset + * + * This function is used to sense on the touch chip, which means to start running the + * FW. The process begin with wakeup the IC bus interface, then write a flag to the IC + * register to make MCU restart running the FW. When sw_reset is true, the function will + * send a command to the IC to leave safe mode. Otherwise, the function will call + * himax_mcu_ic_reset() to reset the touch chip by hardware pin. + * Then enable the HW CRC to protect sram data, and reload to active to make the MCU + * start running without reload the firmware. + * + * Return: 0 on success, negative error code on failure + */ +static int hx83102j_sense_on(struct himax_ts_data *ts, bool sw_reset) +{ + int ret; + const union himax_dword_data re_init = { + .dword = cpu_to_le32(HIMAX_REG_DATA_FW_RE_INIT) + }; + union himax_dword_data data; + + dev_info(ts->dev, "%s: software reset %s\n", __func__, sw_reset ? "true" : "false"); + ret = himax_mcu_interface_on(ts); + if (ret < 0) + return ret; + + ret = himax_mcu_register_write(ts, HIMAX_REG_ADDR_CTRL_FW, re_init.byte, 4); + if (ret < 0) + return ret; + usleep_range(10000, 11000); + if (!sw_reset) { + himax_mcu_ic_reset(ts, false); + } else { + data.word[0] = cpu_to_le16(HIMAX_AHB_CMD_LEAVE_SAFE_MODE); + ret = himax_write(ts, HIMAX_AHB_ADDR_PSW_LB, NULL, data.byte, 2); + if (ret < 0) + return ret; + } + ret = hx83102j_en_hw_crc(ts, true); + if (ret < 0) + return ret; + ret = hx83102j_reload_to_active(ts); + if (ret < 0) + return ret; + + return 0; +} + /** * hx83102j_chip_detect() - Check if the touch chip is HX83102J * @ts: Himax touch screen data @@ -783,166 +919,980 @@ static int hx83102j_read_event_stack(struct himax_ts_data *ts, u8 *buf, u32 leng int ret; const u32 max_trunk_sz = ts->spi_xfer_max_sz - HIMAX_BUS_R_HLEN; - for (i = 0; i < length; i += max_trunk_sz) { - ret = himax_read(ts, HIMAX_AHB_ADDR_EVENT_STACK, buf + i, - min(length - i, max_trunk_sz)); + for (i = 0; i < length; i += max_trunk_sz) { + ret = himax_read(ts, HIMAX_AHB_ADDR_EVENT_STACK, buf + i, + min(length - i, max_trunk_sz)); + if (ret) { + dev_err(ts->dev, "%s: read event stack error!\n", __func__); + return ret; + } + } + + return 0; +} + +/** + * hx83102j_chip_init_data() - Initialize the touch chip data + * @ts: Himax touch screen data + * + * This function is used to initialize hx83102j touch specific data in himax_ts_data. + * The chip_max_dsram_size is the maximum size of the DSRAM of hx83102j. + * + * Return: None + */ +static void hx83102j_chip_init_data(struct himax_ts_data *ts) +{ + ts->chip_max_dsram_size = HIMAX_HX83102J_DSRAM_SZ; +} + +/** + * himax_touch_get() - Get touch data from touch chip + * @ts: Himax touch screen data + * @buf: Buffer to store the data + * + * This function is a wrapper to call hx83102j_read_event_stack() to read the touch + * data from the touch chip. The touch_data_sz is the size of the touch data to read, + * which is calculated by hid report descriptor provided by the firmware. + * + * Return: HIMAX_TS_SUCCESS on success, negative error code on failure. We categorize + * the error code into HIMAX_TS_GET_DATA_FAIL when the read fails, and HIMAX_TS_SUCCESS + * when the read is successful. The reason is that the may need special handling when + * the read fails. + */ +static int himax_touch_get(struct himax_ts_data *ts, u8 *buf) +{ + if (hx83102j_read_event_stack(ts, buf, ts->touch_data_sz)) { + dev_err(ts->dev, "can't read data from chip!"); + return HIMAX_TS_GET_DATA_FAIL; + } + + return HIMAX_TS_SUCCESS; +} + +/** + * himax_mcu_assign_sorting_mode() - Write sorting mode to dsram and verify + * @ts: Himax touch screen data + * @tmp_data_in: password to write + * + * This function is used to write the sorting mode password to dsram and verify the + * password is written correctly. The sorting mode password is used as a flag to + * FW to let it know which mode the touch chip is working on. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_assign_sorting_mode(struct himax_ts_data *ts, u8 *tmp_data_in) +{ + int ret; + u8 rdata[4]; + u32 retry_cnt; + const u32 retry_limit = 3; + + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + ret = himax_mcu_register_write(ts, HIMAX_DSRAM_ADDR_SORTING_MODE_EN, + tmp_data_in, HIMAX_REG_SZ); + if (ret < 0) { + dev_err(ts->dev, "%s: write sorting mode fail\n", __func__); + return ret; + } + usleep_range(1000, 1100); + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_SORTING_MODE_EN, + rdata, HIMAX_REG_SZ); + if (ret < 0) { + dev_err(ts->dev, "%s: read sorting mode fail\n", __func__); + return ret; + } + + if (!memcmp(tmp_data_in, rdata, HIMAX_REG_SZ)) + return 0; + } + dev_err(ts->dev, "%s: fail to write sorting mode\n", __func__); + + return -EINVAL; +} + +/** + * himax_mcu_read_FW_status() - Read FW status from touch chip + * @ts: Himax touch screen data + * + * This function is used to read the FW status from the touch chip. The FW status is + * values from dsram and register from TPIC. Which shows the FW vital working status + * for developer debug. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_read_FW_status(struct himax_ts_data *ts) +{ + int i; + int ret; + size_t len; + u8 data[4]; + const char * const reg_name[] = { + "DBG_MSG", + "FW_STATUS", + "DD_STATUS", + "RESET_FLAG" + }; + const u32 dbg_reg_array[] = { + HIMAX_DSRAM_ADDR_DBG_MSG, + HIMAX_REG_ADDR_FW_STATUS, + HIMAX_REG_ADDR_DD_STATUS, + HIMAX_REG_ADDR_RESET_FLAG + }; + + len = ARRAY_SIZE(dbg_reg_array); + + for (i = 0; i < len; i++) { + ret = himax_mcu_register_read(ts, dbg_reg_array[i], data, HIMAX_REG_SZ); + if (ret < 0) { + dev_err(ts->dev, "%s: read FW status fail\n", __func__); + return ret; + } + + dev_info(ts->dev, "%s: %10s(0x%08X) = 0x%02X, 0x%02X, 0x%02X, 0x%02X\n", + __func__, reg_name[i], dbg_reg_array[i], + data[0], data[1], data[2], data[3]); + } + + return 0; +} + +/** + * himax_mcu_power_on_init() - Power on initialization + * @ts: Himax touch screen data + * + * This function is used to do the power on initialization after firmware has been + * loaded to sram. The process initialize varies IC register and dsram to make sure + * FW start running correctly. When all set, sense on the touch chip to make the FW + * start running and wait for the FW reload done password. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_power_on_init(struct himax_ts_data *ts) +{ + int ret; + u32 retry_cnt; + const u32 retry_limit = 30; + union himax_dword_data data; + + /* RawOut select initial */ + data.dword = cpu_to_le32(HIMAX_DATA_CLEAR); + ret = himax_mcu_register_write(ts, HIMAX_HX83102J_DSRAM_ADDR_RAW_OUT_SEL, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: set RawOut select fail\n", __func__); + return ret; + } + /* Initial sorting mode password to normal mode */ + ret = himax_mcu_assign_sorting_mode(ts, data.byte); + if (ret < 0) { + dev_err(ts->dev, "%s: assign sorting mode fail\n", __func__); + return ret; + } + /* N frame initial */ + /* reset N frame back to default value 1 for normal mode */ + data.dword = cpu_to_le32(1); + ret = himax_mcu_register_write(ts, HIMAX_DSRAM_ADDR_SET_NFRAME, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: set N frame fail\n", __func__); + return ret; + } + /* Initial FW reload status */ + data.dword = cpu_to_le32(HIMAX_DATA_CLEAR); + ret = himax_mcu_register_write(ts, HIMAX_DSRAM_ADDR_2ND_FLASH_RELOAD, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: initial FW reload status fail\n", __func__); + return ret; + } + + ret = hx83102j_sense_on(ts, false); + if (ret < 0) { + dev_err(ts->dev, "%s: sense on fail\n", __func__); + return ret; + } + + dev_info(ts->dev, "%s: waiting for FW reload data\n", __func__); + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_2ND_FLASH_RELOAD, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read FW reload status fail\n", __func__); + return ret; + } + + /* use all 4 bytes to compare */ + if (le32_to_cpu(data.dword) == HIMAX_DSRAM_DATA_FW_RELOAD_DONE) { + dev_info(ts->dev, "%s: FW reload done\n", __func__); + break; + } + dev_info(ts->dev, "%s: wait FW reload %u times\n", __func__, retry_cnt); + ret = himax_mcu_read_FW_status(ts); + if (ret < 0) + dev_err(ts->dev, "%s: read FW status fail\n", __func__); + + usleep_range(10000, 11000); + } + + if (retry_cnt == retry_limit) { + dev_err(ts->dev, "%s: FW reload fail!\n", __func__); + return -EINVAL; + } + + return 0; +} + +/** + * himax_mcu_calculate_crc() - Calculate CRC-32 of given data + * @data: Data to calculate CRC + * @len: Length of data + * + * This function is used to calculate the CRC-32 of the given data. The function + * calculate the CRC-32 value by the polynomial 0x82f63b78. + * + * Return: CRC-32 value + */ +static u32 himax_mcu_calculate_crc(const u8 *data, int len) +{ + int i, j, length; + u32 crc = GENMASK(31, 0); + u32 current_data; + u32 tmp; + const u32 mask = GENMASK(30, 0); + + length = len / 4; + + for (i = 0; i < length; i++) { + current_data = data[i * 4]; + + for (j = 1; j < 4; j++) { + tmp = data[i * 4 + j]; + current_data += (tmp) << (8 * j); + } + crc = current_data ^ crc; + for (j = 0; j < 32; j++) { + if ((crc % 2) != 0) + crc = ((crc >> 1) & mask) ^ CRC32C_POLY_LE; + else + crc = (((crc >> 1) & mask)); + } + } + + return crc; +} + +/** + * himax_mcu_check_crc() - Let TPIC check CRC itself + * @ts: Himax touch screen data + * @start_addr: Start address of the data in sram to check + * @reload_length: Length of the data to check + * @crc_result: CRC result for return + * + * This function is used to let TPIC check the CRC of the given data in sram. The + * function write the start address and length of the data to the TPIC, and wait for + * the TPIC to finish the CRC check. When the CRC check is done, the function read + * the CRC result from the TPIC. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_check_crc(struct himax_ts_data *ts, u32 start_addr, + int reload_length, u32 *crc_result) +{ + int ret; + int length = reload_length / HIMAX_REG_SZ; + u32 retry_cnt; + const u32 retry_limit = 100; + union himax_dword_data data, addr; + + addr.dword = cpu_to_le32(start_addr); + ret = himax_mcu_register_write(ts, HIMAX_REG_ADDR_RELOAD_ADDR_FROM, addr.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: write reload start address fail\n", __func__); + return ret; + } + + data.word[1] = cpu_to_le16(HIMAX_REG_DATA_RELOAD_PASSWORD); + data.word[0] = cpu_to_le16(length); + ret = himax_mcu_register_write(ts, HIMAX_REG_ADDR_RELOAD_ADDR_CMD_BEAT, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: write reload length and password fail!\n", __func__); + return ret; + } + + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_RELOAD_ADDR_CMD_BEAT, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read reload length and password fail!\n", __func__); + return ret; + } + + if (le16_to_cpu(data.word[0]) != length) { + dev_err(ts->dev, "%s: length verify failed!\n", __func__); + return -EINVAL; + } + + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_RELOAD_STATUS, data.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read reload status fail!\n", __func__); + return ret; + } + + data.dword = le32_to_cpu(data.dword); + if ((data.byte[0] & HIMAX_REG_DATA_RELOAD_DONE) != HIMAX_REG_DATA_RELOAD_DONE) { + ret = himax_mcu_register_read(ts, HIMAX_REG_ADDR_RELOAD_CRC32_RESULT, + data.byte, HIMAX_REG_SZ); + if (ret < 0) { + dev_err(ts->dev, "%s: read crc32 result fail!\n", __func__); + return ret; + } + *crc_result = le32_to_cpu(data.dword); + return 0; + } + + dev_info(ts->dev, "%s: Waiting for HW ready!\n", __func__); + usleep_range(1000, 1100); + } + + if (retry_cnt == retry_limit) { + ret = himax_mcu_read_FW_status(ts); + if (ret < 0) + dev_err(ts->dev, "%s: read FW status fail\n", __func__); + } + + return -EINVAL; +} + +/** + * himax_mcu_read_FW_ver() - Read varies version from touch chip + * @ts: Himax touch screen data + * + * This function is used to read the firmware version, config version, touch config + * version, display config version, customer ID, customer info, and project info from + * the touch chip. The function will call himax_mcu_register_read() to read the data + * from the TPIC, and store the data to the IC data in himax_ts_data. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_read_FW_ver(struct himax_ts_data *ts) +{ + int ret; + u8 data[HIMAX_TP_INFO_STR_LEN]; + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_FW_VER, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read FW version fail\n", __func__); + return ret; + } + ts->ic_data.vendor_panel_ver = data[0]; + ts->ic_data.vendor_fw_ver = data[1] << 8 | data[2]; + dev_info(ts->dev, "%s: PANEL_VER: %X\n", __func__, ts->ic_data.vendor_panel_ver); + dev_info(ts->dev, "%s: FW_VER: %X\n", __func__, ts->ic_data.vendor_fw_ver); + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_CFG, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read CFG version fail\n", __func__); + return ret; + } + ts->ic_data.vendor_config_ver = data[2] << 8 | data[3]; + ts->ic_data.vendor_touch_cfg_ver = data[2]; + dev_info(ts->dev, "%s: TOUCH_VER: %X\n", __func__, ts->ic_data.vendor_touch_cfg_ver); + ts->ic_data.vendor_display_cfg_ver = data[3]; + dev_info(ts->dev, "%s: DISPLAY_VER: %X\n", __func__, ts->ic_data.vendor_display_cfg_ver); + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_VENDOR, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read customer ID fail\n", __func__); + return ret; + } + ts->ic_data.vendor_cid_maj_ver = data[2]; + ts->ic_data.vendor_cid_min_ver = data[3]; + dev_info(ts->dev, "%s: CID_VER: %X\n", __func__, (ts->ic_data.vendor_cid_maj_ver << 8 + | ts->ic_data.vendor_cid_min_ver)); + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_CUS_INFO, data, HIMAX_TP_INFO_STR_LEN); + if (ret < 0) { + dev_err(ts->dev, "%s: read customer info fail\n", __func__); + return ret; + } + memcpy(ts->ic_data.vendor_cus_info, data, HIMAX_TP_INFO_STR_LEN); + dev_info(ts->dev, "%s: Cusomer ID : %s\n", __func__, ts->ic_data.vendor_cus_info); + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_PROJ_INFO, data, HIMAX_TP_INFO_STR_LEN); + if (ret < 0) { + dev_err(ts->dev, "%s: read project info fail\n", __func__); + return ret; + } + memcpy(ts->ic_data.vendor_proj_info, data, HIMAX_TP_INFO_STR_LEN); + dev_info(ts->dev, "%s: Project ID : %s\n", __func__, ts->ic_data.vendor_proj_info); + + return 0; +} + +/** + * himax_bin_desc_data_get() - Parse descriptor data from firmware token + * @ts: Himax touch screen data + * @addr: Address of the data in firmware image + * @descript_buf: token for parsing + * + * This function is used to parse the descriptor data from the firmware token. The + * descriptors are mappings of information in the firmware image. The function will + * check checksum of each token first, and then parse the token to get the related + * data. The data includes CID version, FW version, CFG version, touch config table, + * HID table, HID descriptor, and HID read descriptor. + * + * Return: true on success, false on failure + */ +static bool himax_bin_desc_data_get(struct himax_ts_data *ts, u32 addr, u8 *descript_buf) +{ + u16 chk_end; + u16 chk_sum; + u32 hid_table_addr; + u32 i, j; + u32 image_offset; + u32 map_code; + const u32 data_sz = 16; + const u32 report_desc_offset = 24; + union { + u8 *buf; + u32 *word; + } map_data; + + /* looking for mapping in page, each mapping is 16 bytes */ + for (i = 0; i < HIMAX_HX83102J_PAGE_SIZE; i = i + data_sz) { + chk_end = 0; + chk_sum = 0; + for (j = i; j < (i + data_sz); j++) { + chk_end |= descript_buf[j]; + chk_sum += descript_buf[j]; + } + if (!chk_end) { /* 1. Check all zero */ + return false; + } else if (chk_sum % 0x100) { /* 2. Check sum */ + dev_warn(ts->dev, "%s: chk sum failed in %X\n", __func__, i + addr); + } else { /* 3. get data */ + map_data.buf = &descript_buf[i]; + map_code = le32_to_cpup(map_data.word); + map_data.buf = &descript_buf[i + 4]; + image_offset = le32_to_cpup(map_data.word); + /* 4. load info from FW image by specified mapping offset */ + switch (map_code) { + /* Config ID */ + case HIMAX_FW_CID: + ts->fw_info_table.addr_cid_ver_major = image_offset; + ts->fw_info_table.addr_cid_ver_minor = image_offset + 1; + break; + /* FW version */ + case HIMAX_FW_VER: + ts->fw_info_table.addr_fw_ver_major = image_offset; + ts->fw_info_table.addr_fw_ver_minor = image_offset + 1; + break; + /* Config version */ + case HIMAX_CFG_VER: + ts->fw_info_table.addr_cfg_ver_major = image_offset; + ts->fw_info_table.addr_cfg_ver_minor = image_offset + 1; + break; + /* Touch config table */ + case HIMAX_TP_CONFIG_TABLE: + ts->fw_info_table.addr_cfg_table = image_offset; + break; + /* HID table */ + case HIMAX_HID_TABLE: + ts->fw_info_table.addr_hid_table = image_offset; + hid_table_addr = image_offset; + ts->fw_info_table.addr_hid_desc = hid_table_addr; + ts->fw_info_table.addr_hid_rd_desc = + hid_table_addr + report_desc_offset; + break; + } + } + } + + return true; +} + +/** + * himax_mcu_bin_desc_get() - Check and get the bin description from the data + * @fw: Firmware data + * @ts: Himax touch screen data + * @max_sz: Maximum size to check + * + * This function is used to check and get the bin description from the firmware data. + * It will check the given data to see if it match the bin description format, and + * call himax_bin_desc_data_get() to get the related data. + * + * Return: true on mapping_count > 0, false on otherwise + */ +static bool himax_mcu_bin_desc_get(unsigned char *fw, struct himax_ts_data *ts, u32 max_sz) +{ + bool keep_on_flag; + u32 addr; + u32 mapping_count; + unsigned char *fw_buf; + const u8 header_id = 0x87; + const u8 header_id_loc = 0x0e; + const u8 header_sz = 8; + const u8 header[8] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 + }; + + /* Check bin is with description table or not */ + if (!(memcmp(fw, header, header_sz) == 0 && fw[header_id_loc] == header_id)) { + dev_err(ts->dev, "%s: No description table\n", __func__); + return false; + } + + for (addr = 0, mapping_count = 0; addr < max_sz; addr += HIMAX_HX83102J_PAGE_SIZE) { + fw_buf = &fw[addr]; + /* Get related data */ + keep_on_flag = himax_bin_desc_data_get(ts, addr, fw_buf); + if (keep_on_flag) + mapping_count++; + else + break; + } + + return mapping_count > 0; +} + +/** + * himax_mcu_tp_info_check() - Read touch information from touch chip + * @ts: Himax touch screen data + * + * This function is used to read the touch information from the touch chip. The + * information includes the touch resolution, touch point number, interrupt type, + * button number, stylus function, stylus version, and stylus ratio. These information + * is filled by FW after the FW initialized, so it must be called after FW finish + * loading. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_mcu_tp_info_check(struct himax_ts_data *ts) +{ + int ret; + char data[HIMAX_REG_SZ]; + u8 stylus_ratio; + u32 button_num; + u32 max_pt; + u32 rx_num; + u32 tx_num; + u32 x_res; + u32 y_res; + const u32 button_num_mask = 0x03; + const u32 interrupt_type_mask = 0x01; + const u32 interrupt_type_edge = 0x01; + bool int_is_edge; + bool stylus_func; + bool stylus_id_v2; + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_RXNUM_TXNUM, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read rx/tx num fail\n", __func__); + return ret; + } + rx_num = data[2]; + tx_num = data[3]; + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_MAXPT_XYRVS, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read max touch point fail\n", __func__); + return ret; + } + max_pt = data[0]; + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_X_Y_RES, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read x/y resolution fail\n", __func__); + return ret; + } + y_res = be16_to_cpup((u16 *)&data[0]); + x_res = be16_to_cpup((u16 *)&data[2]); + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_INT_IS_EDGE, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read interrupt type fail\n", __func__); + return ret; + } + if ((data[1] & interrupt_type_mask) == interrupt_type_edge) + int_is_edge = true; + else + int_is_edge = false; + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_MKEY, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read button number fail\n", __func__); + return ret; + } + button_num = data[0] & button_num_mask; + + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_STYLUS_FUNCTION, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read stylus function fail\n", __func__); + return ret; + } + stylus_func = data[3] ? true : false; + + if (stylus_func) { + ret = himax_mcu_register_read(ts, HIMAX_DSRAM_ADDR_STYLUS_VERSION, data, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: read stylus version fail\n", __func__); + return ret; + } + /* dsram_addr_stylus_version + 2 : 0=off 1=on */ + stylus_id_v2 = data[2] ? true : false; + /* dsram_addr_stylus_version + 3 : 0=ratio_1 10=ratio_10 */ + stylus_ratio = data[3]; + } + + ts->ic_data.button_num = button_num; + ts->ic_data.stylus_function = stylus_func; + ts->ic_data.rx_num = rx_num; + ts->ic_data.tx_num = tx_num; + ts->ic_data.x_res = x_res; + ts->ic_data.y_res = y_res; + ts->ic_data.max_point = max_pt; + ts->ic_data.interrupt_is_edge = int_is_edge; + if (stylus_func) { + ts->ic_data.stylus_v2 = stylus_id_v2; + ts->ic_data.stylus_ratio = stylus_ratio; + } else { + ts->ic_data.stylus_v2 = false; + ts->ic_data.stylus_ratio = 0; + } + + dev_info(ts->dev, "%s: rx_num = %u, tx_num = %u\n", __func__, + ts->ic_data.rx_num, ts->ic_data.tx_num); + dev_info(ts->dev, "%s: max_point = %u\n", __func__, ts->ic_data.max_point); + dev_info(ts->dev, "%s: interrupt_is_edge = %s, stylus_function = %s\n", __func__, + ts->ic_data.interrupt_is_edge ? "true" : "false", + ts->ic_data.stylus_function ? "true" : "false"); + dev_info(ts->dev, "%s: stylus_v2 = %s, stylus_ratio = %u\n", __func__, + ts->ic_data.stylus_v2 ? "true" : "false", ts->ic_data.stylus_ratio); + dev_info(ts->dev, "%s: TOUCH INFO updated\n", __func__); + + return 0; +} + +/** + * himax_disable_fw_reload() - Disable the FW reload data from flash + * @ts: Himax touch screen data + * + * This function is used to tell FW not to reload data from flash. It needs to be + * set before FW start running. + * + * return: 0 on success, negative error code on failure + */ +static int himax_disable_fw_reload(struct himax_ts_data *ts) +{ + union himax_dword_data data = { + /* + * HIMAX_DSRAM_ADDR_FLASH_RELOAD: 0x10007f00 + * 0x10007f00 <= 0x9aa9, let FW know there's no flash + * <= 0x5aa5, there has flash, but not reload + * <= 0x0000, there has flash, and reload + */ + .dword = cpu_to_le32(HIMAX_DSRAM_DATA_DISABLE_FLASH_RELOAD) + }; + + /* Disable Flash Reload */ + return himax_mcu_register_write(ts, HIMAX_DSRAM_ADDR_FLASH_RELOAD, data.byte, 4); +} + +/** + * himax_sram_write_crc_check() - Write the data to SRAM and check the CRC by hardware + * @ts: Himax touch screen data + * @addr: Address to write to + * @data: Data to write + * @len: Length of data + * + * This function is use to write FW code/data to SRAM and check the CRC by hardware to make + * sure the written data is correct. The FW code is designed to be CRC result 0, so if the + * CRC result is not 0, it means the written data is not correct. + * + * return: 0 on success, negative error code on failure + */ +static int himax_sram_write_crc_check(struct himax_ts_data *ts, u32 addr, const u8 *data, u32 len) +{ + int ret; + u32 crc; + u32 retry_cnt; + const u32 retry_limit = 3; + + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + dev_info(ts->dev, "%s: Write FW to SRAM - total write size = %u\n", __func__, len); + ret = himax_mcu_register_write(ts, addr, data, len); + if (ret) { + dev_err(ts->dev, "%s: write FW to SRAM fail\n", __func__); + return ret; + } + ret = himax_mcu_check_crc(ts, addr, len, &crc); + if (ret) { + dev_err(ts->dev, "%s: check CRC fail\n", __func__); + return ret; + } + dev_info(ts->dev, "%s: HW CRC %s in %u time\n", __func__, + crc == 0 ? "OK" : "Fail", retry_cnt); + + if (crc == 0) + break; + } + + if (crc != 0) { + dev_err(ts->dev, "%s: HW CRC fail\n", __func__); + return -EINVAL; + } + + return 0; +} + +/** + * himax_zf_part_info() - Get and write the partition from the firmware to SRAM + * @fw: Firmware data + * @ts: Himax touch screen data + * + * This function is used to get the partition information from the firmware and write + * the partition to SRAM. The partition information includes the DSRAM address, the + * firmware offset, and the write size. The function will get the partition information + * into a table, and then write the partition to SRAM according to the table. After + * writing the partition to SRAM, the function will check the CRC by hardware to make + * sure the written data is correct. + * + * return: 0 on success, negative error code on failure + */ +static int himax_zf_part_info(const struct firmware *fw, struct himax_ts_data *ts) +{ + int i; + int i_max = -1; + int i_min = -1; + int pnum; + int ret; + u8 buf[HIMAX_ZF_PARTITION_DESC_SZ]; + u32 cfg_crc_sw; + u32 cfg_crc_hw; + u32 cfg_sz; + u32 dsram_base = 0xffffffff; + u32 dsram_max = 0; + u32 retry_cnt = 0; + u32 sram_min; + const u32 retry_limit = 3; + const u32 table_addr = ts->fw_info_table.addr_cfg_table; + struct himax_zf_info *info; + + /* 1. initial check */ + ret = hx83102j_en_hw_crc(ts, true); + if (ret < 0) { + dev_err(ts->dev, "%s: Failed to enable HW CRC\n", __func__); + return ret; + } + pnum = fw->data[table_addr + HIMAX_ZF_PARTITION_AMOUNT_OFFSET]; + if (pnum < 2) { + dev_err(ts->dev, "%s: partition number is not correct\n", __func__); + return -EINVAL; + } + + info = kcalloc(pnum, sizeof(struct himax_zf_info), GFP_KERNEL); + if (!info) + return -ENOMEM; + + memset(info, 0, pnum * sizeof(struct himax_zf_info)); + + /* + * 2. record partition information: + * partition 0: FW main code + */ + memcpy(buf, &fw->data[table_addr], HIMAX_ZF_PARTITION_DESC_SZ); + memcpy(info[0].sram_addr, buf, 4); + info[0].write_size = le32_to_cpup((u32 *)&buf[4]); + info[0].fw_addr = le32_to_cpup((u32 *)&buf[8]); + + /* partition 1 ~ n: config data */ + for (i = 1; i < pnum; i++) { + memcpy(buf, &fw->data[i * HIMAX_ZF_PARTITION_DESC_SZ + table_addr], + HIMAX_ZF_PARTITION_DESC_SZ); + memcpy(info[i].sram_addr, buf, 4); + info[i].write_size = le32_to_cpup((u32 *)&buf[4]); + info[i].fw_addr = le32_to_cpup((u32 *)&buf[8]); + info[i].cfg_addr = le32_to_cpup((u32 *)&info[i].sram_addr[0]); + + /* Write address must be multiple of 4 */ + if (info[i].cfg_addr % 4 != 0) { + info[i].cfg_addr -= (info[i].cfg_addr % 4); + info[i].fw_addr -= (info[i].cfg_addr % 4); + info[i].write_size += (info[i].cfg_addr % 4); + } + + if (dsram_base > info[i].cfg_addr) { + dsram_base = info[i].cfg_addr; + i_min = i; + } + if (dsram_max < info[i].cfg_addr) { + dsram_max = info[i].cfg_addr; + i_max = i; + } + } + + if (i_min < 0 || i_max < 0) { + dev_err(ts->dev, "%s: DSRAM address invalid!\n", __func__); + return -EINVAL; + } + + /* 3. prepare data to update */ + sram_min = info[i_min].cfg_addr; + + cfg_sz = (dsram_max - dsram_base) + info[i_max].write_size; + /* Wrtie size must be multiple of 4 */ + if (cfg_sz % 4 != 0) + cfg_sz = cfg_sz + 4 - (cfg_sz % 4); + + dev_info(ts->dev, "%s: main code sz = %d, config sz = %d\n", __func__, + info[0].write_size, cfg_sz); + /* config size should be smaller than DSRAM size */ + if (cfg_sz > ts->chip_max_dsram_size) { + dev_err(ts->dev, "%s: config size error[%d, %u]!!\n", __func__, + cfg_sz, ts->chip_max_dsram_size); + ret = -EINVAL; + goto alloc_cfg_buffer_failed; + } + + memset(ts->zf_update_cfg_buffer, 0x00, + ts->chip_max_dsram_size * sizeof(u8)); + + /* Collect all partition in FW for DSRAM in a cfg buffer */ + for (i = 1; i < pnum; i++) + memcpy(&ts->zf_update_cfg_buffer[info[i].cfg_addr - dsram_base], + &fw->data[info[i].fw_addr], info[i].write_size); + + /* + * 4. write to sram + * First, write FW main code and check CRC by HW + */ + ret = himax_sram_write_crc_check(ts, le32_to_cpup((u32 *)info[0].sram_addr), + &fw->data[info[0].fw_addr], info[0].write_size); + if (ret < 0) { + dev_err(ts->dev, "%s: HW CRC fail\n", __func__); + goto write_main_code_failed; + } + + /* + * Second, FW config data: Calculate CRC of CFG data which is going to write. + * CFG data don't have CRC pre-defined in FW and need to be calculated by driver. + */ + cfg_crc_sw = himax_mcu_calculate_crc(ts->zf_update_cfg_buffer, cfg_sz); + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + /* Write hole cfg data to DSRAM */ + dev_info(ts->dev, "%s: Write cfg to SRAM - total write size = %d\n", + __func__, cfg_sz); + ret = himax_mcu_register_write(ts, sram_min, ts->zf_update_cfg_buffer, cfg_sz); + if (ret < 0) { + dev_err(ts->dev, "%s: write cfg to SRAM fail\n", __func__); + goto write_cfg_failed; + } + /* + * Check CRC: Tell HW to calculate CRC from CFG start address in SRAM and check + * size is equal to size of CFG buffer written. Then we compare the two CRC data + * make sure data written is correct. + */ + ret = himax_mcu_check_crc(ts, sram_min, cfg_sz, &cfg_crc_hw); if (ret) { - dev_err(ts->dev, "%s: read event stack error!\n", __func__); - return ret; + dev_err(ts->dev, "%s: check CRC failed!\n", __func__); + goto crc_failed; } - } - return 0; -} + if (cfg_crc_hw != cfg_crc_sw) + dev_err(ts->dev, "%s: Cfg CRC FAIL, HWCRC = %X, SWCRC = %X, retry = %u\n", + __func__, cfg_crc_hw, cfg_crc_sw, retry_cnt); + else + break; + } -/** - * himax_touch_get() - Get touch data from touch chip - * @ts: Himax touch screen data - * @buf: Buffer to store the data - * - * This function is a wrapper to call hx83102j_read_event_stack() to read the touch - * data from the touch chip. The touch_data_sz is the size of the touch data to read, - * which is calculated by hid report descriptor provided by the firmware. - * - * Return: HIMAX_TS_SUCCESS on success, negative error code on failure. We categorize - * the error code into HIMAX_TS_GET_DATA_FAIL when the read fails, and HIMAX_TS_SUCCESS - * when the read is successful. The reason is that the may need special handling when - * the read fails. - */ -static int himax_touch_get(struct himax_ts_data *ts, u8 *buf) -{ - if (hx83102j_read_event_stack(ts, buf, ts->touch_data_sz)) { - dev_err(ts->dev, "can't read data from chip!"); - return HIMAX_TS_GET_DATA_FAIL; + if (retry_cnt == retry_limit && cfg_crc_hw != cfg_crc_sw) { + dev_err(ts->dev, "%s: Write cfg to SRAM fail\n", __func__); + ret = -EINVAL; + goto crc_not_match; } - return HIMAX_TS_SUCCESS; +crc_not_match: +crc_failed: +write_cfg_failed: +write_main_code_failed: +alloc_cfg_buffer_failed: + kfree(info); + + return ret; } /** - * himax_bin_desc_data_get() - Parse descriptor data from firmware token + * himax_mcu_firmware_update_zf() - Update the firmware to the touch chip + * @fw: Firmware data * @ts: Himax touch screen data - * @addr: Address of the data in firmware image - * @descript_buf: token for parsing * - * This function is used to parse the descriptor data from the firmware token. The - * descriptors are mappings of information in the firmware image. The function will - * check checksum of each token first, and then parse the token to get the related - * data. The data includes CID version, FW version, CFG version, touch config table, - * HID table, HID descriptor, and HID read descriptor. + * This function is used to update the firmware to the touch chip. The first step is + * to reset the touch chip, stop the MCU and then write the firmware to the touch chip. * - * Return: true on success, false on failure + * return: 0 on success, negative error code on failure */ -static bool himax_bin_desc_data_get(struct himax_ts_data *ts, u32 addr, u8 *descript_buf) +static int himax_mcu_firmware_update_zf(const struct firmware *fw, struct himax_ts_data *ts) { - u16 chk_end; - u16 chk_sum; - u32 hid_table_addr; - u32 i, j; - u32 image_offset; - u32 map_code; - const u32 data_sz = 16; - const u32 report_desc_offset = 24; - union { - u8 *buf; - u32 *word; - } map_data; + int ret; + union himax_dword_data data_system_reset = { + .dword = cpu_to_le32(HIMAX_REG_DATA_SYSTEM_RESET) + }; - /* looking for mapping in page, each mapping is 16 bytes */ - for (i = 0; i < HIMAX_HX83102J_PAGE_SIZE; i = i + data_sz) { - chk_end = 0; - chk_sum = 0; - for (j = i; j < (i + data_sz); j++) { - chk_end |= descript_buf[j]; - chk_sum += descript_buf[j]; - } - if (!chk_end) { /* 1. Check all zero */ - return false; - } else if (chk_sum % 0x100) { /* 2. Check sum */ - dev_warn(ts->dev, "%s: chk sum failed in %X\n", __func__, i + addr); - } else { /* 3. get data */ - map_data.buf = &descript_buf[i]; - map_code = le32_to_cpup(map_data.word); - map_data.buf = &descript_buf[i + 4]; - image_offset = le32_to_cpup(map_data.word); - /* 4. load info from FW image by specified mapping offset */ - switch (map_code) { - /* Config ID */ - case HIMAX_FW_CID: - ts->fw_info_table.addr_cid_ver_major = image_offset; - ts->fw_info_table.addr_cid_ver_minor = image_offset + 1; - break; - /* FW version */ - case HIMAX_FW_VER: - ts->fw_info_table.addr_fw_ver_major = image_offset; - ts->fw_info_table.addr_fw_ver_minor = image_offset + 1; - break; - /* Config version */ - case HIMAX_CFG_VER: - ts->fw_info_table.addr_cfg_ver_major = image_offset; - ts->fw_info_table.addr_cfg_ver_minor = image_offset + 1; - break; - /* Touch config table */ - case HIMAX_TP_CONFIG_TABLE: - ts->fw_info_table.addr_cfg_table = image_offset; - break; - /* HID table */ - case HIMAX_HID_TABLE: - ts->fw_info_table.addr_hid_table = image_offset; - hid_table_addr = image_offset; - ts->fw_info_table.addr_hid_desc = hid_table_addr; - ts->fw_info_table.addr_hid_rd_desc = - hid_table_addr + report_desc_offset; - break; - } - } + dev_info(ts->dev, "%s: Updating FW - total FW size = %u\n", __func__, (u32)fw->size); + ret = himax_mcu_register_write(ts, HIMAX_REG_ADDR_SYSTEM_RESET, data_system_reset.byte, 4); + if (ret < 0) { + dev_err(ts->dev, "%s: system reset fail\n", __func__); + return ret; } - return true; + ret = hx83102j_sense_off(ts, false); + if (ret) + return ret; + + ret = himax_zf_part_info(fw, ts); + + return ret; } /** - * himax_mcu_bin_desc_get() - Check and get the bin description from the data - * @fw: Firmware data + * himax_zf_reload_from_file() - Complete firmware update sequence + * @file_name: File name of the firmware * @ts: Himax touch screen data - * @max_sz: Maximum size to check * - * This function is used to check and get the bin description from the firmware data. - * It will check the given data to see if it match the bin description format, and - * call himax_bin_desc_data_get() to get the related data. + * This function process the full sequence of updating the firmware to the touch chip. + * It will first check if the other thread is updating now, if not, it will request the + * firmware from user space and then call himax_mcu_firmware_update_zf() to update the + * firmware, and then tell firmware not to reload data from flash and initial the touch + * chip by calling himax_mcu_power_on_init(). * - * Return: true on mapping_count > 0, false on otherwise + * return: 0 on success, negative error code on failure */ -static bool himax_mcu_bin_desc_get(unsigned char *fw, struct himax_ts_data *ts, u32 max_sz) +static int himax_zf_reload_from_file(char *file_name, struct himax_ts_data *ts) { - bool keep_on_flag; - u32 addr; - u32 mapping_count; - unsigned char *fw_buf; - const u8 header_id = 0x87; - const u8 header_id_loc = 0x0e; - const u8 header_sz = 8; - const u8 header[8] = { - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 - }; + int ret; + const struct firmware *fw; - /* Check bin is with description table or not */ - if (!(memcmp(fw, header, header_sz) == 0 && fw[header_id_loc] == header_id)) { - dev_err(ts->dev, "%s: No description table\n", __func__); - return false; + if (!mutex_trylock(&ts->zf_update_lock)) { + dev_warn(ts->dev, "%s: Other thread is updating now!\n", __func__); + return 0; } + dev_info(ts->dev, "%s: Preparing to update %s!\n", __func__, file_name); - for (addr = 0, mapping_count = 0; addr < max_sz; addr += HIMAX_HX83102J_PAGE_SIZE) { - fw_buf = &fw[addr]; - /* Get related data */ - keep_on_flag = himax_bin_desc_data_get(ts, addr, fw_buf); - if (keep_on_flag) - mapping_count++; - else - break; + ret = request_firmware(&fw, file_name, ts->dev); + if (ret < 0) { + dev_err(ts->dev, "%s: request firmware fail, code[%d]!!\n", __func__, ret); + goto load_firmware_error; } - return mapping_count > 0; + ret = himax_mcu_firmware_update_zf(fw, ts); + release_firmware(fw); + if (ret < 0) + goto load_firmware_error; + + ret = himax_disable_fw_reload(ts); + if (ret < 0) + goto load_firmware_error; + ret = himax_mcu_power_on_init(ts); + +load_firmware_error: + mutex_unlock(&ts->zf_update_lock); + + return ret; } /** @@ -1238,6 +2188,7 @@ static int himax_ts_operation(struct himax_ts_data *ts) * This function is used to handle interrupt bottom half work. It will * call the himax_ts_operation() to get the touch data, dispatch the data * to HID core. If the touch data is not valid, it will reset the TPIC. + * It will also call the hx83102j_reload_to_active() after the reset action. * * Return: void */ @@ -1246,7 +2197,38 @@ static void himax_ts_work(struct himax_ts_data *ts) if (himax_ts_operation(ts) == HIMAX_TS_GET_DATA_FAIL) { dev_info(ts->dev, "%s: Now reset the Touch chip\n", __func__); himax_mcu_ic_reset(ts, true); + if (hx83102j_reload_to_active(ts)) + dev_warn(ts->dev, "%s: Reload to active failed\n", __func__); + } +} + +/** + * himax_update_fw() - update firmware using firmware structure + * @ts: Himax touch screen data + * + * This function use already initialize firmware structure in ts to update + * firmware. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_update_fw(struct himax_ts_data *ts) +{ + int ret; + u32 retry_cnt; + const u32 retry_limit = 3; + + for (retry_cnt = 0; retry_cnt < retry_limit; retry_cnt++) { + ret = himax_mcu_firmware_update_zf(ts->himax_fw, ts); + if (ret < 0) { + dev_err(ts->dev, "%s: TP upgrade error, upgrade_times = %d\n", __func__, + retry_cnt); + } else { + dev_info(ts->dev, "%s: TP FW upgrade OK\n", __func__); + return 0; + } } + + return -EIO; } /** @@ -1279,9 +2261,8 @@ static int himax_hid_rd_init(struct himax_ts_data *ts) if (!ts->hid_rd_data.rd_data) return -ENOMEM; } - /* Copy the base RD from firmware table */ memcpy((void *)ts->hid_rd_data.rd_data, - &ts->himax_fw_data[ts->fw_info_table.addr_hid_rd_desc], + &ts->himax_fw->data[ts->fw_info_table.addr_hid_rd_desc], ts->hid_desc.report_desc_length); ts->hid_rd_data.rd_length = ts->hid_desc.report_desc_length; } @@ -1343,81 +2324,105 @@ static int himax_hid_report_data_init(struct himax_ts_data *ts) return 0; } -/* load firmware data from flash, parse HID info and register HID */ /** - * himax_load_config() - Load the firmware from the flash - * @ts: Himax touch screen data + * himax_initial_work() - Initial work for the touch screen + * @work: Work structure + * + * This function is used to do the initial work for the touch screen. It will + * call the request_firmware() to get the firmware from the file system, and parse the + * mapping table in 1k header. If the headers are parsed successfully, it will + * call the himax_update_fw() to update the firmware and power on the touch screen. + * If the power on action is successful, it will load the hid descriptor and + * check the touch panel information. If the touch panel information is correct, + * it will call the himax_hid_rd_init() to initialize the HID report descriptor, + * and call the himax_hid_register() to register the HID device. After all is done, + * it will release the firmware and enable the interrupt. * - * This function is used to load the firmware from the flash. It will read - * the firmware from the flash and parse the HID info. If the HID info is - * valid, it will initialize the HID report descriptor and register the HID - * device. If the HID device is probed, it will initialize the report data - * and enable the interrupt. - * - * Return: 0 on success, negative error code on failure + * Return: None */ -static int himax_load_config(struct himax_ts_data *ts) +static void himax_initial_work(struct work_struct *work) { + struct himax_ts_data *ts = container_of(work, struct himax_ts_data, + initial_work.work); int ret; - s32 i; - s32 page_sz = (s32)HIMAX_HX83102J_PAGE_SIZE; - s32 flash_sz = (s32)HIMAX_HX83102J_FLASH_SIZE; - bool fw_load_status = false; + bool fw_load_status; const u32 fw_bin_header_sz = 1024; ts->ic_boot_done = false; - - ts->himax_fw_data = devm_kzalloc(ts->dev, HIMAX_HX83102J_FLASH_SIZE, GFP_KERNEL); - if (!ts->himax_fw_data) - return -ENOMEM; - - for (i = 0; i < flash_sz; i += page_sz) { - ret = himax_mcu_register_read(ts, i, ts->himax_fw_data + i, - (flash_sz - i) > page_sz ? page_sz : (flash_sz - i)); - if (ret < 0) { - dev_err(ts->dev, "%s: read FW from flash fail!\n", __func__); - return ret; - } + dev_info(ts->dev, "%s: request file %s\n", __func__, ts->firmware_name); + ret = request_firmware(&ts->himax_fw, ts->firmware_name, ts->dev); + if (ret < 0) { + dev_err(ts->dev, "%s: request firmware failed, error code = %d\n", __func__, ret); + return; } - /* Search mapping table in 1k header */ - fw_load_status = himax_mcu_bin_desc_get((unsigned char *)ts->himax_fw_data, + /* Parse the mapping table in 1k header */ + fw_load_status = himax_mcu_bin_desc_get((unsigned char *)ts->himax_fw->data, ts, fw_bin_header_sz); if (!fw_load_status) { - dev_err(ts->dev, "%s: FW load status fail!\n", __func__); - return -EINVAL; + dev_err(ts->dev, "%s: Failed to parse the mapping table!\n", __func__); + goto err_load_bin_descriptor; } - if (ts->fw_info_table.addr_hid_desc != 0) { - memcpy(&ts->hid_desc, - &ts->himax_fw_data[ts->fw_info_table.addr_hid_desc], - sizeof(struct himax_hid_desc)); - ts->hid_desc.desc_length = - le16_to_cpu(ts->hid_desc.desc_length); - ts->hid_desc.bcd_version = - le16_to_cpu(ts->hid_desc.bcd_version); - ts->hid_desc.report_desc_length = - le16_to_cpu(ts->hid_desc.report_desc_length); - ts->hid_desc.max_input_length = - le16_to_cpu(ts->hid_desc.max_input_length); - ts->hid_desc.max_output_length = - le16_to_cpu(ts->hid_desc.max_output_length); - ts->hid_desc.max_fragment_length = - le16_to_cpu(ts->hid_desc.max_fragment_length); - ts->hid_desc.vendor_id = - le16_to_cpu(ts->hid_desc.vendor_id); - ts->hid_desc.product_id = - le16_to_cpu(ts->hid_desc.product_id); - ts->hid_desc.version_id = - le16_to_cpu(ts->hid_desc.version_id); - ts->hid_desc.flags = - le16_to_cpu(ts->hid_desc.flags); + if (himax_update_fw(ts)) { + dev_err(ts->dev, "%s: Update FW fail\n", __func__); + goto err_update_fw_failed; } + dev_info(ts->dev, "%s: Update FW success\n", __func__); + /* write flag to sram to stop fw reload again. */ + if (himax_disable_fw_reload(ts)) + goto err_disable_fw_reload; + if (himax_mcu_power_on_init(ts)) + goto err_power_on_init; + /* get hid descriptors */ + if (!ts->fw_info_table.addr_hid_desc) { + dev_err(ts->dev, "%s: No HID descriptor! Wrong FW!\n", __func__); + goto err_wrong_firmware; + } + memcpy(&ts->hid_desc, + &ts->himax_fw->data[ts->fw_info_table.addr_hid_desc], + sizeof(struct himax_hid_desc)); + ts->hid_desc.desc_length = + le16_to_cpu(ts->hid_desc.desc_length); + ts->hid_desc.bcd_version = + le16_to_cpu(ts->hid_desc.bcd_version); + ts->hid_desc.report_desc_length = + le16_to_cpu(ts->hid_desc.report_desc_length); + ts->hid_desc.max_input_length = + le16_to_cpu(ts->hid_desc.max_input_length); + ts->hid_desc.max_output_length = + le16_to_cpu(ts->hid_desc.max_output_length); + ts->hid_desc.max_fragment_length = + le16_to_cpu(ts->hid_desc.max_fragment_length); + ts->hid_desc.vendor_id = + le16_to_cpu(ts->hid_desc.vendor_id); + ts->hid_desc.product_id = + le16_to_cpu(ts->hid_desc.product_id); + ts->hid_desc.version_id = + le16_to_cpu(ts->hid_desc.version_id); + ts->hid_desc.flags = + le16_to_cpu(ts->hid_desc.flags); + + if (himax_mcu_tp_info_check(ts)) + goto err_tp_info_failed; + if (himax_mcu_read_FW_ver(ts)) + goto err_read_fw_ver; + if (ts->pdata.pid) { + if (ts->pdata.pid != ts->hid_desc.product_id) { + dev_err(ts->dev, "%s: PID mismatch, dtsi PID = 0x%x, fw PID = 0x%x\n", + __func__, ts->pdata.pid, ts->hid_desc.product_id); + goto err_pid_match_failed; + } else { + dev_info(ts->dev, "%s: PID match, dtsi PID = 0x%x, fw PID = 0x%x\n", + __func__, ts->pdata.pid, ts->hid_desc.product_id); + } + } if (himax_hid_rd_init(ts)) { dev_err(ts->dev, "%s: hid rd init fail\n", __func__); goto err_hid_rd_init_failed; } + usleep_range(1000000, 1000100); himax_hid_register(ts); if (!ts->hid_probed) { goto err_hid_probe_failed; @@ -1428,19 +2433,29 @@ static int himax_load_config(struct himax_ts_data *ts) } } - ts->himax_fw_data = NULL; + release_firmware(ts->himax_fw); + ts->himax_fw = NULL; + ts->ic_boot_done = true; himax_int_enable(ts, true); - return 0; + return; err_report_data_init_failed: himax_hid_remove(ts); ts->hid_probed = false; err_hid_probe_failed: err_hid_rd_init_failed: - - return -EINVAL; +err_pid_match_failed: +err_read_fw_ver: +err_tp_info_failed: +err_wrong_firmware: +err_power_on_init: +err_disable_fw_reload: +err_update_fw_failed: +err_load_bin_descriptor: + release_firmware(ts->himax_fw); + ts->himax_fw = NULL; } /** @@ -1492,12 +2507,22 @@ static void himax_ap_notify_fw_suspend(struct himax_ts_data *ts, bool suspend) * @ts: Himax touch screen data * * This function is used to resume the touch screen. It will call the + * himax_zf_reload_from_file() to reload the firmware. And call the * himax_ap_notify_fw_suspend() to notify the FW of AP resume status. * * Return: None */ static void himax_resume_proc(struct himax_ts_data *ts) { + int ret; + + ret = himax_zf_reload_from_file(ts->firmware_name, ts); + if (ret) { + dev_err(ts->dev, "%s: update FW fail, code[%d]!!\n", __func__, ret); + return; + } + ts->resume_succeeded = true; + himax_ap_notify_fw_suspend(ts, false); } @@ -1527,15 +2552,23 @@ static int himax_chip_suspend(struct himax_ts_data *ts) * This function is used to resume the touch screen. It will set the resume * success flag to false, and disable reset pin. Then call the himax_resume_proc() * to process detailed resume procedure. + * If the resume action is succeeded, it will call the himax_hid_probe() to restore + * the HID device and enable the interrupt. * * Return: 0 on success, negative error code on failure */ static int himax_chip_resume(struct himax_ts_data *ts) { + ts->resume_succeeded = false; gpiod_set_value(ts->pdata.gpiod_rst, 0); himax_resume_proc(ts); - himax_hid_probe(ts); - himax_int_enable(ts, true); + if (ts->resume_succeeded) { + himax_hid_probe(ts); + himax_int_enable(ts, true); + } else { + dev_err(ts->dev, "%s: resume failed!\n", __func__); + return -ECANCELED; + } return 0; } @@ -1596,8 +2629,7 @@ static int himax_resume(struct device *dev) * initialize interrupt lock, register the interrupt, and disable the * interrupt. If later part of initialization succeed, then interrupt will * be enabled. - * It will also load the firmware from the flash, parse the HID info, and - * register the HID device by calling the himax_load_config(). + * And initialize varies flags, workqueue and delayed work for later use. * * Return: 0 on success, negative error code on failure */ @@ -1605,18 +2637,96 @@ static int himax_chip_init(struct himax_ts_data *ts) { int ret; + hx83102j_chip_init_data(ts); if (himax_ts_register_interrupt(ts)) { dev_err(ts->dev, "%s: register interrupt failed\n", __func__); return -EIO; } himax_int_enable(ts, false); - ret = himax_load_config(ts); - if (ret < 0) - return ret; + ts->zf_update_cfg_buffer = devm_kzalloc(ts->dev, ts->chip_max_dsram_size, GFP_KERNEL); + if (!ts->zf_update_cfg_buffer) { + ret = -ENOMEM; + goto err_update_cfg_buf_alloc_failed; + } + INIT_DELAYED_WORK(&ts->initial_work, himax_initial_work); + schedule_delayed_work(&ts->initial_work, msecs_to_jiffies(HIMAX_DELAY_BOOT_UPDATE_MS)); ts->initialized = true; + return 0; + cancel_delayed_work_sync(&ts->initial_work); +err_update_cfg_buf_alloc_failed: + + return ret; +} + +/** + * himax_chip_deinit() - Deinitialize the Himax touch screen + * @ts: Himax touch screen data + * + * This function is used to deinitialize the Himax touch screen. + * + * Return: None + */ +static void himax_chip_deinit(struct himax_ts_data *ts) +{ + cancel_delayed_work_sync(&ts->initial_work); +} + +#if defined(CONFIG_OF) +/** + * himax_parse_dt() - Parse the device tree + * @dt: Device node + * @pdata: Himax platform data + * + * This function is used to parse the device tree. If "himax,pid" is found, + * it will parse the PID value and set it to the platform data. The firmware + * name will set to himax_i2chid_$PID.bin if the PID is found, or + * himax_i2chid.bin if the PID is not found. + * + * Return: 0 on success, negative error code on failure + */ +static int himax_parse_dt(struct device_node *dt, struct himax_platform_data *pdata) +{ + u32 data; + const char default_fw_name[] = HIMAX_BOOT_UPGRADE_FWNAME; + /* + * Maximum length of a firmware name size: + * (default_name) + _XXXX(PID) + .bin + null terminator + */ + static char pid_fw_name[ARRAY_SIZE(default_fw_name) + 5 + 4 + 1] = {0}; + struct himax_ts_data *ts; + + if (!dt || !pdata) + return -EINVAL; + + ts = container_of(pdata, struct himax_ts_data, pdata); + /* Set default firmware name, without PID */ + strscpy(ts->firmware_name, HIMAX_BOOT_UPGRADE_FWNAME HIMAX_FW_EXT_NAME, + sizeof(HIMAX_BOOT_UPGRADE_FWNAME HIMAX_FW_EXT_NAME)); + + /* + * check himax,pid first, if exist then get the value. + * himax,pid = <0x1002>; 0x1002 is PID value + */ + if (of_get_property(dt, "himax,pid", &data)) { + if (of_property_read_u32(dt, "himax,pid", &data)) { + pdata->pid = 0; + return -EINVAL; + } + + pdata->pid = data; + snprintf(pid_fw_name, sizeof(pid_fw_name), "%s_%04X%s", HIMAX_BOOT_UPGRADE_FWNAME, + pdata->pid, HIMAX_FW_EXT_NAME); + dev_info(ts->dev, "%s: DT:himax,pid = %04X, fw_name = %s\n", __func__, + pdata->pid, pid_fw_name); + strscpy(ts->firmware_name, pid_fw_name, sizeof(pid_fw_name)); + } else { + pdata->pid = 0; + } + return 0; } +#endif /** * __himax_initial_power_up() - Initial power up of the Himax touch screen @@ -1820,6 +2930,13 @@ static int himax_spi_drv_probe(struct spi_device *spi) dev_err(ts->dev, "%s: gpio-rst value is not valid\n", __func__); return -EIO; } +#if defined(CONFIG_OF) + if (himax_parse_dt(spi->dev.of_node, pdata) < 0) { + dev_err(ts->dev, "%s: parse OF data failed!\n", __func__); + ts->dev = NULL; + return -ENODEV; + } +#endif spi->bits_per_word = 8; spi->mode = SPI_MODE_3; @@ -1850,6 +2967,7 @@ static int himax_spi_drv_probe(struct spi_device *spi) spin_lock_init(&ts->irq_lock); mutex_init(&ts->rw_lock); mutex_init(&ts->reg_lock); + mutex_init(&ts->zf_update_lock); dev_set_drvdata(&spi->dev, ts); spi_set_drvdata(spi, ts); @@ -1885,6 +3003,7 @@ static void himax_spi_drv_remove(struct spi_device *spi) if (ts->hid_probed) himax_hid_remove(ts); } + himax_chip_deinit(ts); himax_platform_deinit(ts); } } diff --git a/drivers/hid/hid-himax-83102j.h b/drivers/hid/hid-himax-83102j.h index eef55c45b1d4..6a41ff680478 100644 --- a/drivers/hid/hid-himax-83102j.h +++ b/drivers/hid/hid-himax-83102j.h @@ -10,7 +10,9 @@ // #define HX_PWR_CONFIG #include +#include #include +#include #include #include #include @@ -41,6 +43,13 @@ HIMAX_BUS_W_HLEN + HIMAX_REG_SZ) /* SPI CS setup time */ #define HIMAX_SPI_CS_SETUP_TIME 300 +/* Clear 4 bytes data */ +#define HIMAX_DATA_CLEAR 0x00000000 +/* boot update start delay */ +#define HIMAX_DELAY_BOOT_UPDATE_MS 2000 +#define HIMAX_TP_INFO_STR_LEN 12U +#define HIMAX_ZF_PARTITION_AMOUNT_OFFSET 12 +#define HIMAX_ZF_PARTITION_DESC_SZ 16U /* HIDRAW report header size */ #define HIMAX_HID_REPORT_HDR_SZ 2U /* hx83102j IC parameters */ @@ -71,17 +80,48 @@ #define HIMAX_AHB_CMD_INCR4_ADD_4_BYTE 0x01 #define HIMAX_AHB_CMD_LEAVE_SAFE_MODE 0x0000 /* DSRAM flag addresses */ +#define HIMAX_DSRAM_ADDR_VENDOR 0x10007000 +#define HIMAX_DSRAM_ADDR_FW_VER 0x10007004 +#define HIMAX_DSRAM_ADDR_CUS_INFO 0x10007008 +#define HIMAX_DSRAM_ADDR_PROJ_INFO 0x10007014 +#define HIMAX_DSRAM_ADDR_CFG 0x10007084 +#define HIMAX_DSRAM_ADDR_INT_IS_EDGE 0x10007088 +#define HIMAX_DSRAM_ADDR_MKEY 0x100070e8 +#define HIMAX_DSRAM_ADDR_RXNUM_TXNUM 0x100070f4 +#define HIMAX_DSRAM_ADDR_MAXPT_XYRVS 0x100070f8 +#define HIMAX_DSRAM_ADDR_X_Y_RES 0x100070fc +#define HIMAX_DSRAM_ADDR_STYLUS_FUNCTION 0x1000719c +#define HIMAX_DSRAM_ADDR_STYLUS_VERSION 0x100071fc +#define HIMAX_DSRAM_ADDR_SET_NFRAME 0x10007294 +#define HIMAX_DSRAM_ADDR_2ND_FLASH_RELOAD 0x100072c0 +#define HIMAX_DSRAM_ADDR_FLASH_RELOAD 0x10007f00 +#define HIMAX_DSRAM_ADDR_SORTING_MODE_EN 0x10007f04 +#define HIMAX_DSRAM_ADDR_DBG_MSG 0x10007f40 #define HIMAX_DSRAM_ADDR_AP_NOTIFY_FW_SUSPEND 0x10007fd0 /* dsram flag data */ #define HIMAX_DSRAM_DATA_AP_NOTIFY_FW_SUSPEND 0xa55aa55a #define HIMAX_DSRAM_DATA_AP_NOTIFY_FW_RESUME 0x00000000 +#define HIMAX_DSRAM_DATA_DISABLE_FLASH_RELOAD 0x00009aa9 +#define HIMAX_DSRAM_DATA_FW_RELOAD_DONE 0x000072c0 /* hx83102j-specific register/dsram flags/data */ +#define HIMAX_HX83102J_DSRAM_ADDR_RAW_OUT_SEL 0x100072ec +#define HIMAX_HX83102J_REG_ADDR_HW_CRC 0x80010000 #define HIMAX_HX83102J_REG_ADDR_TCON_RST 0x80020004 +#define HIMAX_HX83102J_REG_DATA_HW_CRC 0x0000ecce +#define HIMAX_HX83102J_REG_DATA_HW_CRC_DISABLE 0x00000000 /* hardware register addresses */ #define HIMAX_REG_ADDR_SPI200_DATA 0x8000002c +#define HIMAX_REG_ADDR_RELOAD_STATUS 0x80050000 +#define HIMAX_REG_ADDR_RELOAD_CRC32_RESULT 0x80050018 +#define HIMAX_REG_ADDR_RELOAD_ADDR_FROM 0x80050020 +#define HIMAX_REG_ADDR_RELOAD_ADDR_CMD_BEAT 0x80050028 +#define HIMAX_REG_ADDR_SYSTEM_RESET 0x90000018 +#define HIMAX_REG_ADDR_RELOAD_TO_ACTIVE 0x90000048 #define HIMAX_REG_ADDR_CTRL_FW 0x9000005c #define HIMAX_REG_ADDR_FW_STATUS 0x900000a8 #define HIMAX_REG_ADDR_ICID 0x900000d0 +#define HIMAX_REG_ADDR_RESET_FLAG 0x900000e4 +#define HIMAX_REG_ADDR_DD_STATUS 0x900000e8 /* hardware reg data/flags */ #define HIMAX_REG_DATA_FW_STATE_RUNNING 0x05 #define HIMAX_REG_DATA_FW_STATE_SAFE_MODE 0x0c @@ -89,6 +129,9 @@ #define HIMAX_REG_DATA_FW_GO_SAFEMODE 0xa5 #define HIMAX_REG_DATA_FW_IN_SAFEMODE 0x87 #define HIMAX_REG_DATA_ICID 0x83102900 +#define HIMAX_REG_DATA_RELOAD_DONE 0x01 +#define HIMAX_REG_DATA_RELOAD_PASSWORD 0x99 +#define HIMAX_REG_DATA_SYSTEM_RESET 0x00000055 #define HIMAX_REG_DATA_TCON_RST 0x00000000 /* HIMAX SPI function select, 1st byte of any SPI command sequence */ #define HIMAX_SPI_FUNCTION_READ 0xf3 @@ -100,6 +143,8 @@ #define HIMAX_CFG_VER 0x10000600 #define HIMAX_HID_TABLE 0x30000100 #define HIMAX_FW_BIN_DESC 0x10000000 +#define HIMAX_BOOT_UPGRADE_FWNAME "himax_i2chid" +#define HIMAX_FW_EXT_NAME ".bin" /** * enum himax_hidraw_id_function - HIDRAW report IDs @@ -119,6 +164,20 @@ enum himax_touch_report_status { HIMAX_TS_SUCCESS = 0, }; +/** + * struct himax_zf_info - Zero flash update information + * @sram_addr: SRAM address byte array buffer + * @write_size: Write size of each chunk + * @fw_addr: Offset in firmware file + * @cfg_addr: target sram address + */ +struct himax_zf_info { + u8 sram_addr[4]; + int write_size; + u32 fw_addr; + u32 cfg_addr; +}; + /** * struct himax_fw_address_table - address/offset in firmware image * @addr_fw_ver_major: Address to Major version of firmware @@ -170,9 +229,21 @@ union himax_dword_data { /** * struct himax_ic_data - IC information holder * @stylus_ratio: Stylus ratio + * @vendor_cus_info: Vendor customer information + * @vendor_proj_info: Vendor project information + * @vendor_fw_ver: Vendor firmware version + * @vendor_config_ver: Vendor config version + * @vendor_touch_cfg_ver: Vendor touch config version + * @vendor_display_cfg_ver: Vendor display config version + * @vendor_cid_maj_ver: Vendor CID major version + * @vendor_cid_min_ver: Vendor CID minor version + * @vendor_panel_ver: Vendor panel version + * @vendor_sensor_id: Vendor sensor ID * @rx_num: Number of RX * @tx_num: Number of TX * @button_num: Number of buttons + * @x_res: X resolution + * @y_res: Y resolution * @max_point: Maximum touch point * @icid: IC ID * @interrupt_is_edge: Interrupt is edge otherwise level @@ -181,9 +252,21 @@ union himax_dword_data { */ struct himax_ic_data { u8 stylus_ratio; + u8 vendor_cus_info[12]; + u8 vendor_proj_info[12]; + int vendor_fw_ver; + int vendor_config_ver; + int vendor_touch_cfg_ver; + int vendor_display_cfg_ver; + int vendor_cid_maj_ver; + int vendor_cid_min_ver; + int vendor_panel_ver; + int vendor_sensor_id; u32 rx_num; u32 tx_num; u32 button_num; + u32 x_res; + u32 y_res; u32 max_point; u32 icid; bool interrupt_is_edge; @@ -191,6 +274,38 @@ struct himax_ic_data { bool stylus_v2; }; +/** + * struct himax_bin_desc - Firmware binary descriptor + * @passwd: Password to indicate the binary is valid + * @cid: Customer ID + * @panel_ver: Panel version + * @fw_ver: Firmware version + * @ic_sign: IC signature + * @customer: Customer name + * @project: Project name + * @fw_major: Major version of firmware + * @fw_minor: Minor version of firmware + * @date: Generate date of firmware + * @ic_sign_2: IC signature 2 + * + * This structure is used to hold the firmware binary descriptor. + * It directly maps to a sequence of bytes in firmware image, + * thus need to be packed. + */ +struct himax_bin_desc { + u16 passwd; + u16 cid; + u8 panel_ver; + u16 fw_ver; + u8 ic_sign; + char customer[12]; + char project[12]; + char fw_major[12]; + char fw_minor[12]; + char date[12]; + char ic_sign_2[12]; +} __packed; + /** * struct himax_hid_desc - HID descriptor * @desc_length: Length of HID descriptor @@ -223,8 +338,42 @@ struct himax_hid_desc { u32 reserved; } __packed; +/** + * struct himax_hid_info - IC information holder for HIDRAW function + * @vid: Vendor ID + * @pid: Product ID + * @cfg_info: Configuration information + * @cfg_version: Configuration version + * @disp_version: Display version + * @rx: Number of RX + * @tx: Number of TX + * @y_res: Y resolution + * @x_res: X resolution + * @pt_num: Number of touch points + * @mkey_num: Number of mkey + * @debug_info: Debug information + * + * This structure is used to hold the IC config information for HIDRAW. + * The format is binary fixed, thus need to be packed. + */ +struct himax_hid_info { + u16 vid; + u16 pid; + u8 cfg_info[32]; + u8 cfg_version; + u8 disp_version; + u8 rx; + u8 tx; + u16 y_res; + u16 x_res; + u8 pt_num; + u8 mkey_num; + u8 debug_info[78]; +} __packed; + /** * struct himax_platform_data - Platform data holder + * @pid: Product ID * @is_panel_follower: Is panel follower enabled * @panel_follower: DRM panel follower * @gpiod_rst: GPIO reset @@ -232,6 +381,7 @@ struct himax_hid_desc { * This structure is used to hold the platform related data. */ struct himax_platform_data { + u16 pid; bool is_panel_follower; struct drm_panel_follower panel_follower; struct gpio_desc *gpiod_rst; @@ -242,8 +392,9 @@ struct himax_platform_data { * @xfer_buf: Interrupt data buffer * @xfer_rx_data: SPI Transfer receive data buffer * @xfer_tx_data: SPI Transfer transmit data buffer - * @himax_fw_data: Firmware data holder from flash + * @zf_update_cfg_buffer: Zero flash update configuration buffer * @himax_irq: IRQ number + * @chip_max_dsram_size: Maximum size of DSRAM * @spi_xfer_max_sz: Size of SPI controller max transfer size * @xfer_buf_sz: Size of interrupt data buffer * @irq_state: IRQ state @@ -252,24 +403,30 @@ struct himax_platform_data { * @probe_finish: Indicate the driver probe is finished * @ic_boot_done: Indicate the IC boot is done * @hid_probed: Indicate the HID device is probed + * @resume_succeeded: Indicate the resume is succeeded + * @firmware_name: Firmware name * @touch_data_sz: Size of each interrupt data from IC + * @himax_fw: Firmware data holder from user space * @dev: Device pointer * @spi: SPI device pointer * @hid: HID device pointer * @reg_lock: Mutex lock for reg access * @rw_lock: Mutex lock for read/write action + * @zf_update_lock: Mutex lock for zero-flash FW update * @ic_data: IC information holder * @pdata: Platform data holder * @fw_info_table: Firmware information address table of firmware image * @hid_desc: HID descriptor * @hid_rd_data: HID report descriptor data + * @initial_work: Delayed work for TP initialization */ struct himax_ts_data { u8 *xfer_buf; u8 *xfer_rx_data; u8 *xfer_tx_data; - u8 *himax_fw_data; + u8 *zf_update_cfg_buffer; s32 himax_irq; + u32 chip_max_dsram_size; u32 spi_xfer_max_sz; u32 xfer_buf_sz; atomic_t irq_state; @@ -279,7 +436,11 @@ struct himax_ts_data { bool probe_finish; bool ic_boot_done; bool hid_probed; + bool resume_succeeded; + bool zf_update_flag; + char firmware_name[64]; int touch_data_sz; + const struct firmware *himax_fw; struct device *dev; struct spi_device *spi; struct hid_device *hid; @@ -287,10 +448,13 @@ struct himax_ts_data { struct mutex reg_lock; /* lock for bus read/write action */ struct mutex rw_lock; + /* lock for zero-flash FW update */ + struct mutex zf_update_lock; struct himax_ic_data ic_data; struct himax_platform_data pdata; struct himax_fw_address_table fw_info_table; struct himax_hid_desc hid_desc; struct himax_hid_rd_data hid_rd_data; + struct delayed_work initial_work; }; #endif