From patchwork Thu Jan 16 15:01:46 2025 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Guillaume Ranquet X-Patchwork-Id: 13941823 Received: from mail-wm1-f50.google.com (mail-wm1-f50.google.com [209.85.128.50]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 46A8E22B590 for ; Thu, 16 Jan 2025 15:06:50 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=209.85.128.50 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1737040014; cv=none; b=UkdImcOThDsB79EasyQNZFy1vjmw+ngxojf/LTrzGueoNwPHPl1rfkLzvI5Ofkgc+abkZcys8WJ9OuR9wSUAGPf62Cjy0ZUvITlCGH0JF+eaJAuP246VhhdLEy4zHG+c6Vmr5bHceMzGH5Vd3Cjl+JhUN4NgZSWbKaPOblXL+14= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1737040014; c=relaxed/simple; bh=OvMvpa5hJvjp4mkxfj0XCsyUfbH0TO5uL7Xwk9BXyfI=; h=From:Date:Subject:MIME-Version:Content-Type:Message-Id:References: In-Reply-To:To:Cc; b=PVGGFSa5vRoKo4zfg0VikJ4JqN9zouoW2UEGEQ1LFtRCh2lSeiVovITXZVhylqT1wrwGx0dY01SGdTsv7MHz2RPvQv+kvVTL1YdtXdC2Vc9F2ECungdFr4HpReFHC7KCFnFHdCyRr4beHu1BPZNqxhKyOkaL4G0FX456flo/kU4= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=none (p=none dis=none) header.from=baylibre.com; spf=pass smtp.mailfrom=baylibre.com; dkim=pass (2048-bit key) header.d=baylibre-com.20230601.gappssmtp.com header.i=@baylibre-com.20230601.gappssmtp.com header.b=KF1uh0hq; arc=none smtp.client-ip=209.85.128.50 Authentication-Results: smtp.subspace.kernel.org; dmarc=none (p=none dis=none) header.from=baylibre.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=baylibre.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=baylibre-com.20230601.gappssmtp.com header.i=@baylibre-com.20230601.gappssmtp.com header.b="KF1uh0hq" Received: by mail-wm1-f50.google.com with SMTP id 5b1f17b1804b1-436a39e4891so6809935e9.1 for ; Thu, 16 Jan 2025 07:06:49 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=baylibre-com.20230601.gappssmtp.com; s=20230601; t=1737040008; x=1737644808; darn=vger.kernel.org; h=cc:to:in-reply-to:references:message-id:content-transfer-encoding :mime-version:subject:date:from:from:to:cc:subject:date:message-id :reply-to; bh=bYmh0880XE2lLymcGq8FMdf3HLknZRQFCBrcBxm7rsc=; b=KF1uh0hqtYe2KxnOqJuJU35WovaY8KY7x+SzmiUQp7SclFS+0l08eO5pi+iOE08D2W BIlMBeWDPqvNOijMum3o60S4xwuIjFwGfFqqTMHGpPx7IZ4FKLop70owvx65JakNXrpE P3u9M6znnOKb4gW2AQFSE1AuiDI7XR3ofbf/u5j52SnfszDAgrPgeUfUvlavZE24G6JW q1vFeL7viEl2xW+guu7VbDjYhn3BfrcNoWP0EBgJMyyQVcKDAL60ZqN+j9dvETV4NZ2W pLVfsTqBG08a7T5QP6EmQzel5g9X+3xbHFuw1wea4Oj1M7d3HEYlhl9l+19CcXDxGJDA H8ww== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1737040008; x=1737644808; h=cc:to:in-reply-to:references:message-id:content-transfer-encoding :mime-version:subject:date:from:x-gm-message-state:from:to:cc :subject:date:message-id:reply-to; bh=bYmh0880XE2lLymcGq8FMdf3HLknZRQFCBrcBxm7rsc=; b=wXtJ72Oo3N9UhL2GwjizwygJOKxj82DOnpSWzufjtaK/WFBQ2IVophmP6kONv796+j AxSX7JgKlZPsQbY+LowYIaQk6BESNTktugvgSjSVNhirBF0XFHQW7zPzJOuzCrhNP4Ai XOFzgy9yFh8hBXyrq5iRlvFzBexcThdx/jf3SMmbY3sIBBLzF/EDaZbbOlpvOXIvLFax 6yDrP5c/pwIbrXeH4Q1ckNY1Jnc2M+LbuhbxBH13leo9BzkVPzrKYaS51K4cYmibFBIp PIDCZP4Z6x9zQ1qhDWClfTbME+ojAbXMbCzzKdPTROj1esctDPruUTyMLlTTP20P7rCM sFTA== X-Gm-Message-State: AOJu0YxuuencemvH+r8c4JVq2kvr+opZEy+Q/5nJ0cHzltl+F9g+OAka 6zMWub1+muw5s9T8P0/ok/H8QJ2qONWKrPMvM4tE9UKyc5T4fbVTVSAlU7ziHH0= X-Gm-Gg: ASbGncva4wjlf+WsAtUIhd5lQMEzpp7IpLfSZ0Mxclkbx/dhIN++rIEUT6YYDfnQZmc agjq/kmPcG5ALBsPxjpnebBXuCVatnGNFV27XdrBqldnXSofKT+UIAnSnC7Wn/XsjZwyBTguLME 8oC8faN04BSC8QOiH5lHjGXkzHpXF/ERtvA8qc7wQy4W0B6dq/GCX/8LtnbiWeMuZFh2T/xj/sZ 65d+6K3Ucda5YC7K6Qo8bugLP7rMJ2Bxv0CJ6I0X7VgAk2EKFgpSjHnMa2c2OqM7/0N9IOmh1o3 W1egcyliprv6YIYCrV3tGg/OKjA= X-Google-Smtp-Source: AGHT+IHU27OYlDGXMzXBaKrU+DkQAHWS8+mLw8V3qTbihM+GrLtuH3Q/whk1TW3oyYUFbrFi53pf9A== X-Received: by 2002:a05:600c:1c28:b0:434:a734:d268 with SMTP id 5b1f17b1804b1-436e2699dfbmr347566565e9.14.1737040008384; Thu, 16 Jan 2025 07:06:48 -0800 (PST) Received: from [127.0.0.1] (alille-653-1-300-114.w90-1.abo.wanadoo.fr. [90.1.180.114]) by smtp.gmail.com with ESMTPSA id 5b1f17b1804b1-43890408a66sm2199445e9.5.2025.01.16.07.06.47 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Thu, 16 Jan 2025 07:06:47 -0800 (PST) From: Guillaume Ranquet Date: Thu, 16 Jan 2025 16:01:46 +0100 Subject: [PATCH RFC v3 1/2] iio: introduce the FAULT event type Precedence: bulk X-Mailing-List: linux-iio@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Message-Id: <20250116-ad4111_openwire-v3-1-ea9ebf29bd1d@baylibre.com> References: <20250116-ad4111_openwire-v3-0-ea9ebf29bd1d@baylibre.com> In-Reply-To: <20250116-ad4111_openwire-v3-0-ea9ebf29bd1d@baylibre.com> To: Lars-Peter Clausen , Michael Hennerich , Jonathan Cameron Cc: linux-iio@vger.kernel.org, linux-kernel@vger.kernel.org, Guillaume Ranquet X-Mailer: b4 0.15-dev Add a new event type to describe an hardware failure. Signed-off-by: Guillaume Ranquet --- drivers/iio/industrialio-event.c | 2 ++ include/uapi/linux/iio/types.h | 2 ++ tools/iio/iio_event_monitor.c | 4 ++++ 3 files changed, 8 insertions(+) diff --git a/drivers/iio/industrialio-event.c b/drivers/iio/industrialio-event.c index db06501b0e61a91e3b06345b418504803f4aefb5..ea3dcdc0b70a7b8d53281d92d9d3bacbc74638ba 100644 --- a/drivers/iio/industrialio-event.c +++ b/drivers/iio/industrialio-event.c @@ -232,6 +232,7 @@ static const char * const iio_ev_type_text[] = { [IIO_EV_TYPE_CHANGE] = "change", [IIO_EV_TYPE_MAG_REFERENCED] = "mag_referenced", [IIO_EV_TYPE_GESTURE] = "gesture", + [IIO_EV_TYPE_FAULT] = "fault", }; static const char * const iio_ev_dir_text[] = { @@ -240,6 +241,7 @@ static const char * const iio_ev_dir_text[] = { [IIO_EV_DIR_FALLING] = "falling", [IIO_EV_DIR_SINGLETAP] = "singletap", [IIO_EV_DIR_DOUBLETAP] = "doubletap", + [IIO_EV_DIR_OPENWIRE] = "openwire", }; static const char * const iio_ev_info_text[] = { diff --git a/include/uapi/linux/iio/types.h b/include/uapi/linux/iio/types.h index 12886d4465e4896aedce837c2df63c78f83a5496..ac0f5ee0bac4aa44577d7d772d3628a084f5c645 100644 --- a/include/uapi/linux/iio/types.h +++ b/include/uapi/linux/iio/types.h @@ -119,6 +119,7 @@ enum iio_event_type { IIO_EV_TYPE_CHANGE, IIO_EV_TYPE_MAG_REFERENCED, IIO_EV_TYPE_GESTURE, + IIO_EV_TYPE_FAULT, }; enum iio_event_direction { @@ -128,6 +129,7 @@ enum iio_event_direction { IIO_EV_DIR_NONE, IIO_EV_DIR_SINGLETAP, IIO_EV_DIR_DOUBLETAP, + IIO_EV_DIR_OPENWIRE, }; #endif /* _UAPI_IIO_TYPES_H_ */ diff --git a/tools/iio/iio_event_monitor.c b/tools/iio/iio_event_monitor.c index cccf62ea2b8f9b55a83a4960c1a60087c7b053f3..efa1807ab505a507f10aa2d314e03e40b71f62ba 100644 --- a/tools/iio/iio_event_monitor.c +++ b/tools/iio/iio_event_monitor.c @@ -75,6 +75,7 @@ static const char * const iio_ev_type_text[] = { [IIO_EV_TYPE_CHANGE] = "change", [IIO_EV_TYPE_MAG_REFERENCED] = "mag_referenced", [IIO_EV_TYPE_GESTURE] = "gesture", + [IIO_EV_TYPE_FAULT] = "fault", }; static const char * const iio_ev_dir_text[] = { @@ -83,6 +84,7 @@ static const char * const iio_ev_dir_text[] = { [IIO_EV_DIR_FALLING] = "falling", [IIO_EV_DIR_SINGLETAP] = "singletap", [IIO_EV_DIR_DOUBLETAP] = "doubletap", + [IIO_EV_DIR_OPENWIRE] = "openwire", }; static const char * const iio_modifier_names[] = { @@ -249,6 +251,7 @@ static bool event_is_known(struct iio_event_data *event) case IIO_EV_TYPE_MAG_ADAPTIVE: case IIO_EV_TYPE_CHANGE: case IIO_EV_TYPE_GESTURE: + case IIO_EV_TYPE_FAULT: break; default: return false; @@ -260,6 +263,7 @@ static bool event_is_known(struct iio_event_data *event) case IIO_EV_DIR_FALLING: case IIO_EV_DIR_SINGLETAP: case IIO_EV_DIR_DOUBLETAP: + case IIO_EV_DIR_OPENWIRE: case IIO_EV_DIR_NONE: break; default: From patchwork Thu Jan 16 15:01:47 2025 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Guillaume Ranquet X-Patchwork-Id: 13941822 Received: from mail-wm1-f44.google.com (mail-wm1-f44.google.com [209.85.128.44]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id 6848F22B5A4 for ; Thu, 16 Jan 2025 15:06:51 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=209.85.128.44 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1737040013; cv=none; b=RaJuzicqGcll0a/qVzO312+JAKcVmNhwA7lp43VEu8jmu9wkYF0Up0aiSqxvdWhpUmlHl4h/Y1Z+GF5AFN1vNddL/4rzZbs+kJtTAMOWO3qQFVCh8P4PTGNa9R4s3dA2QpyXieBvrdqOC7U4nN/ZL2F9vHT3RtjNeyPS4WykbPA= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1737040013; c=relaxed/simple; bh=O4OVg6THaAyJCN31ATnrdwx4XSZqib0B0lm4Mb0xZO0=; h=From:Date:Subject:MIME-Version:Content-Type:Message-Id:References: In-Reply-To:To:Cc; b=k2D8pKCJRl5XtjuEx7aNVkyNSGp4ovhyJDYsdkL2pbl9qZoisYW1Nn/Trab6Oywxq7Vggp+LLNgCpySjzEXGLPm9BJTqXaOOy3Y7tUP99Pg1h+NrgbzMfaArvMjXDGhYVMQJtvOHM1blDPvGQqZWYwQ5uoS2Qys1e7g/K2q2D5s= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=none (p=none dis=none) header.from=baylibre.com; spf=pass smtp.mailfrom=baylibre.com; dkim=pass (2048-bit key) header.d=baylibre-com.20230601.gappssmtp.com header.i=@baylibre-com.20230601.gappssmtp.com header.b=VEk3g5Tz; arc=none smtp.client-ip=209.85.128.44 Authentication-Results: smtp.subspace.kernel.org; dmarc=none (p=none dis=none) header.from=baylibre.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=baylibre.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=baylibre-com.20230601.gappssmtp.com header.i=@baylibre-com.20230601.gappssmtp.com header.b="VEk3g5Tz" Received: by mail-wm1-f44.google.com with SMTP id 5b1f17b1804b1-437a92d7b96so9751625e9.2 for ; Thu, 16 Jan 2025 07:06:51 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=baylibre-com.20230601.gappssmtp.com; s=20230601; t=1737040010; x=1737644810; darn=vger.kernel.org; h=cc:to:in-reply-to:references:message-id:content-transfer-encoding :mime-version:subject:date:from:from:to:cc:subject:date:message-id :reply-to; bh=sjlwA4REiY5R5IhsqMiI8UAgPLeOfi/kkzWK9drJoFU=; b=VEk3g5TzhK/YgQLLdA/QLxwsV3ElhT8Tf7C0p5boFVPRS1yBviNKmkP5w4PAS9ls4x foO/AgD1xKYwGc/P4jC26TQNGCeAS9LA4ef1iBetdVdt+HNTRfKmYanvMs11JCLl53tU PbzOxzRpEXlJazoLI3yUfS5T+/oj3u4teecmav17GejPTRywKW3Fk+hw31plH4iFBHmS CjqMRIDgsjX9A2O3lkfYBqGKuFH8MZd0xir0iAhKaVDUiK90PEvtF9v1/5mbLykEsbVn 14gdCdqZKGlhP/8PWu22RnXUAFkIG+rFInvTdP6ZwqOYFGq6MbbwzSwgDNCsnt7v/5XG msCg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1737040010; x=1737644810; h=cc:to:in-reply-to:references:message-id:content-transfer-encoding :mime-version:subject:date:from:x-gm-message-state:from:to:cc :subject:date:message-id:reply-to; bh=sjlwA4REiY5R5IhsqMiI8UAgPLeOfi/kkzWK9drJoFU=; b=JhCbpI3NTuFLmpI6KmuMICT2hTC0ljLDBWUsDWDdr6xAYURlyfLKW6j2y7Id0GsML0 CNb0JQf/E/YCfSCmNjj09DIr9nWJOPqDdcVn1Lv+yeyKY7wCOcvxo/rod3kZrKX6K/Eb A4bYKRWwNaQnHM5MnSyXDW/ZQYfT06g1DroM5Y7zCOBnrGs0ayNDn93xI4jmsmX/2LHD hvq6TmBNaYVt2AEmPoPiY3+eNTXEv8ekRtSrx5m9zuYpF6No4mj9UAd14xnc/jaQRIwV WjqtcyM7eYgBb1MLWWBdAhTXh1z7aSLZ2GrA6NPZoIrjbsVdpY3XijcKv3KljckA7j0Y yWZA== X-Gm-Message-State: AOJu0Yzr56R/KbQ8bB79/FC7dEc9SETaSVdt61Iu6tB69ICZvOUV8sUE gL03/p3O85FT2ZsFJPkQfSMc9nS/qudrj+wBwB81toJ9hSBDtcSfNd18Askw524= X-Gm-Gg: ASbGncun3dLbde0PH6rBGHItgVS3jH2dr0OERAgSC9X7FNhKMrbUy6agVaLe841ImFt RneNWAfxmLPkyYc5zB73gmVb2eg/AANc9PpXz/0A0jn3xQq12Pdk9eZA8PBWXfWpb7QKB9Nqd/z ASjIQPM4cKV1c00CYPY13v6BUaPesx85yGpqGZf3G3fyvqAdliR+8BZbOiC8EXw09RmHllZhqYL ljib+4qJQsKFuAeh0RROtQ5UtxjPfnq9fygZ0axvK/uWzikza9GVaajfn90za5HYu8+gjtuSXed ejO4UlmJQo80BnDDJOokdQPDieM= X-Google-Smtp-Source: AGHT+IECd8vb1yBNgdVoN0mAFwUUdm8B4OkaaNAv0U6vAjPVVcNNFT/bHjbZZe0FFacoSpI09q6oMQ== X-Received: by 2002:a05:6000:1aca:b0:385:fa3d:1988 with SMTP id ffacd0b85a97d-38a872fc2d2mr29186696f8f.8.1737040009509; Thu, 16 Jan 2025 07:06:49 -0800 (PST) Received: from [127.0.0.1] (alille-653-1-300-114.w90-1.abo.wanadoo.fr. [90.1.180.114]) by smtp.gmail.com with ESMTPSA id 5b1f17b1804b1-43890408a66sm2199445e9.5.2025.01.16.07.06.48 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Thu, 16 Jan 2025 07:06:48 -0800 (PST) From: Guillaume Ranquet Date: Thu, 16 Jan 2025 16:01:47 +0100 Subject: [PATCH RFC v3 2/2] iio: adc: ad7173: add openwire detection support for single conversions Precedence: bulk X-Mailing-List: linux-iio@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Message-Id: <20250116-ad4111_openwire-v3-2-ea9ebf29bd1d@baylibre.com> References: <20250116-ad4111_openwire-v3-0-ea9ebf29bd1d@baylibre.com> In-Reply-To: <20250116-ad4111_openwire-v3-0-ea9ebf29bd1d@baylibre.com> To: Lars-Peter Clausen , Michael Hennerich , Jonathan Cameron Cc: linux-iio@vger.kernel.org, linux-kernel@vger.kernel.org, Guillaume Ranquet X-Mailer: b4 0.15-dev Some chips of the ad7173 family supports open wire detection. Generate a level fault event whenever an external source is disconnected from the system input on single conversions. Signed-off-by: Guillaume Ranquet --- drivers/iio/adc/ad7173.c | 161 +++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 161 insertions(+) diff --git a/drivers/iio/adc/ad7173.c b/drivers/iio/adc/ad7173.c index 11ff148cb5a315d32485acf04b8d6f7d0fb6e5fa..d1cba93722673d2f7fd9239074d36e5274527557 100644 --- a/drivers/iio/adc/ad7173.c +++ b/drivers/iio/adc/ad7173.c @@ -35,6 +35,7 @@ #include #include +#include #include #include #include @@ -102,6 +103,7 @@ #define AD7173_GPIO_PDSW BIT(14) #define AD7173_GPIO_OP_EN2_3 BIT(13) +#define AD4111_GPIO_GP_OW_EN BIT(12) #define AD7173_GPIO_MUX_IO BIT(12) #define AD7173_GPIO_SYNC_EN BIT(11) #define AD7173_GPIO_ERR_EN BIT(10) @@ -149,6 +151,7 @@ #define AD7173_FILTER_ODR0_MASK GENMASK(5, 0) #define AD7173_MAX_CONFIGS 8 +#define AD4111_OW_DET_THRSH_MV 300 #define AD7173_MODE_CAL_INT_ZERO 0x4 /* Internal Zero-Scale Calibration */ #define AD7173_MODE_CAL_INT_FULL 0x5 /* Internal Full-Scale Calibration */ @@ -181,11 +184,15 @@ struct ad7173_device_info { bool has_int_ref; bool has_ref2; bool has_internal_fs_calibration; + bool has_openwire_det; bool higher_gpio_bits; u8 num_gpios; }; struct ad7173_channel_config { + /* Openwire detection threshold */ + unsigned int openwire_thrsh_raw; + int openwire_comp_chan; u8 cfg_slot; bool live; @@ -202,6 +209,7 @@ struct ad7173_channel { unsigned int chan_reg; unsigned int ain; struct ad7173_channel_config cfg; + bool openwire_det_en; }; struct ad7173_state { @@ -280,6 +288,7 @@ static const struct ad7173_device_info ad4111_device_info = { .has_current_inputs = true, .has_int_ref = true, .has_internal_fs_calibration = true, + .has_openwire_det = true, .clock = 2 * HZ_PER_MHZ, .sinc5_data_rates = ad7173_sinc5_data_rates, .num_sinc5_data_rates = ARRAY_SIZE(ad7173_sinc5_data_rates), @@ -616,6 +625,72 @@ static int ad7173_calibrate_all(struct ad7173_state *st, struct iio_dev *indio_d return 0; } +/* + * Associative array of channel pairs for open wire detection + * The array is indexed by ain and gives the associated channel pair + * to perform the open wire detection with + * the channel pair [0] is for non differential and pair [1] + * is for differential inputs + */ +static int openwire_ain_to_channel_pair[][2][2] = { +/* AIN Single Differential */ + [0] = { {0, 15}, {1, 2} }, + [1] = { {1, 2}, {2, 1} }, + [2] = { {3, 4}, {5, 6} }, + [3] = { {5, 6}, {6, 5} }, + [4] = { {7, 8}, {9, 10} }, + [5] = { {9, 10}, {10, 9} }, + [6] = { {11, 12}, {13, 14} }, + [7] = { {13, 14}, {14, 13} }, +}; + +/* + * Openwire detection on ad4111 works by running the same input measurement + * on two different channels and compare if the difference between the two + * measurements exceeds a certain value (typical 300mV) + */ +static int ad4111_openwire_event(struct iio_dev *indio_dev, + const struct iio_chan_spec *chan) +{ + struct ad7173_state *st = iio_priv(indio_dev); + struct ad7173_channel *adchan = &st->channels[chan->address]; + struct ad7173_channel_config *cfg = &adchan->cfg; + int ret, val1, val2; + + ret = regmap_set_bits(st->reg_gpiocon_regmap, AD7173_REG_GPIO, AD4111_GPIO_GP_OW_EN); + if (ret) + return ret; + + adchan->cfg.openwire_comp_chan = + openwire_ain_to_channel_pair[chan->channel][chan->differential][0]; + + ret = ad_sigma_delta_single_conversion(indio_dev, chan, &val1); + if (ret < 0) { + dev_err(&indio_dev->dev, "Error running ad_sigma_delta single conversion: %d", ret); + goto out; + } + + adchan->cfg.openwire_comp_chan = + openwire_ain_to_channel_pair[chan->channel][chan->differential][1]; + + ret = ad_sigma_delta_single_conversion(indio_dev, chan, &val2); + if (ret < 0) { + dev_err(&indio_dev->dev, "Error running ad_sigma_delta single conversion: %d", ret); + goto out; + } + + if (abs(val1 - val2) > cfg->openwire_thrsh_raw) + iio_push_event(indio_dev, + IIO_UNMOD_EVENT_CODE(IIO_VOLTAGE, chan->address, + IIO_EV_TYPE_FAULT, IIO_EV_DIR_OPENWIRE), + iio_get_time_ns(indio_dev)); + +out: + adchan->cfg.openwire_comp_chan = -1; + regmap_clear_bits(st->reg_gpiocon_regmap, AD7173_REG_GPIO, AD4111_GPIO_GP_OW_EN); + return ret; +} + static int ad7173_mask_xlate(struct gpio_regmap *gpio, unsigned int base, unsigned int offset, unsigned int *reg, unsigned int *mask) @@ -813,6 +888,9 @@ static int ad7173_set_channel(struct ad_sigma_delta *sd, unsigned int channel) FIELD_PREP(AD7173_CH_SETUP_SEL_MASK, st->channels[channel].cfg.cfg_slot) | st->channels[channel].ain; + if (st->channels[channel].cfg.openwire_comp_chan >= 0) + channel = st->channels[channel].cfg.openwire_comp_chan; + return ad_sd_write_reg(&st->sd, AD7173_REG_CH(channel), 2, val); } @@ -861,6 +939,11 @@ static int ad7173_disable_all(struct ad_sigma_delta *sd) static int ad7173_disable_one(struct ad_sigma_delta *sd, unsigned int chan) { + struct ad7173_state *st = ad_sigma_delta_to_ad7173(sd); + + if (st->channels[chan].cfg.openwire_comp_chan >= 0) + chan = st->channels[chan].cfg.openwire_comp_chan; + return ad_sd_write_reg(sd, AD7173_REG_CH(chan), 2, 0); } @@ -968,6 +1051,12 @@ static int ad7173_read_raw(struct iio_dev *indio_dev, if (ret < 0) return ret; + if (ch->openwire_det_en) { + ret = ad4111_openwire_event(indio_dev, chan); + if (ret < 0) + return ret; + } + return IIO_VAL_INT; case IIO_CHAN_INFO_SCALE: @@ -1112,12 +1201,58 @@ static int ad7173_debug_reg_access(struct iio_dev *indio_dev, unsigned int reg, return ad_sd_write_reg(&st->sd, reg, reg_size, writeval); } +static int ad7173_write_event_config(struct iio_dev *indio_dev, + const struct iio_chan_spec *chan, + enum iio_event_type type, + enum iio_event_direction dir, + bool state) +{ + struct ad7173_state *st = iio_priv(indio_dev); + struct ad7173_channel *adchan = &st->channels[chan->address]; + + switch (type) { + case IIO_EV_TYPE_FAULT: + adchan->openwire_det_en = state; + default: + return -EINVAL; + } + + return 0; +} + +static int ad7173_read_event_config(struct iio_dev *indio_dev, const struct iio_chan_spec *chan, + enum iio_event_type type, enum iio_event_direction dir) +{ + struct ad7173_state *st = iio_priv(indio_dev); + struct ad7173_channel *adchan = &st->channels[chan->address]; + + switch (type) { + case IIO_EV_TYPE_FAULT: + return adchan->openwire_det_en; + default: + return -EINVAL; + } + + return 0; +} + +static const struct iio_event_spec ad4111_events[] = { + { + .type = IIO_EV_TYPE_FAULT, + .dir = IIO_EV_DIR_OPENWIRE, + .mask_separate = BIT(IIO_EV_INFO_VALUE), + .mask_shared_by_all = BIT(IIO_EV_INFO_ENABLE), + }, +}; + static const struct iio_info ad7173_info = { .read_raw = &ad7173_read_raw, .write_raw = &ad7173_write_raw, .debugfs_reg_access = &ad7173_debug_reg_access, .validate_trigger = ad_sd_validate_trigger, .update_scan_mode = ad7173_update_scan_mode, + .write_event_config = ad7173_write_event_config, + .read_event_config = ad7173_read_event_config, }; static const struct iio_scan_type ad4113_scan_type = { @@ -1321,6 +1456,19 @@ static int ad7173_validate_reference(struct ad7173_state *st, int ref_sel) return 0; } +static int ad7173_validate_openwire_ain_inputs(struct ad7173_state *st, bool differential, + unsigned int ain0, unsigned int ain1) +{ + /* + * If the channel is configured as differential, + * the ad4111 requires specific ains to be used together + */ + if (differential) + return (ain0 % 2) ? (ain0 - 1) == ain1 : (ain0 + 1) == ain1; + + return ain1 == AD4111_VINCOM_INPUT; +} + static int ad7173_fw_parse_channel_config(struct iio_dev *indio_dev) { struct ad7173_channel *chans_st_arr, *chan_st_priv; @@ -1375,6 +1523,7 @@ static int ad7173_fw_parse_channel_config(struct iio_dev *indio_dev) chan_st_priv->cfg.bipolar = false; chan_st_priv->cfg.input_buf = st->info->has_input_buf; chan_st_priv->cfg.ref_sel = AD7173_SETUP_REF_SEL_INT_REF; + chan_st_priv->cfg.openwire_comp_chan = -1; st->adc_mode |= AD7173_ADC_MODE_REF_EN; if (st->info->data_reg_only_16bit) chan_arr[chan_index].scan_type = ad4113_scan_type; @@ -1442,6 +1591,7 @@ static int ad7173_fw_parse_channel_config(struct iio_dev *indio_dev) chan_st_priv->chan_reg = chan_index; chan_st_priv->cfg.input_buf = st->info->has_input_buf; chan_st_priv->cfg.odr = 0; + chan_st_priv->cfg.openwire_comp_chan = -1; chan_st_priv->cfg.bipolar = fwnode_property_read_bool(child, "bipolar"); if (chan_st_priv->cfg.bipolar) @@ -1456,6 +1606,17 @@ static int ad7173_fw_parse_channel_config(struct iio_dev *indio_dev) chan_st_priv->cfg.input_buf = st->info->has_input_buf; chan->channel2 = ain[1]; chan_st_priv->ain = AD7173_CH_ADDRESS(ain[0], ain[1]); + if (st->info->has_openwire_det && + ad7173_validate_openwire_ain_inputs(st, chan->differential, ain[0], ain[1])) { + chan->event_spec = ad4111_events; + chan->num_event_specs = ARRAY_SIZE(ad4111_events); + chan_st_priv->cfg.openwire_thrsh_raw = + BIT(chan->scan_type.realbits - !!(chan_st_priv->cfg.bipolar)) + * AD4111_OW_DET_THRSH_MV + / ad7173_get_ref_voltage_milli(st, chan_st_priv->cfg.ref_sel); + if (chan->channel < st->info->num_voltage_in_div) + chan_st_priv->cfg.openwire_thrsh_raw /= AD4111_DIVIDER_RATIO; + } } if (st->info->data_reg_only_16bit)