Received: by 2002:ac0:98c7:0:0:0:0:0 with SMTP id g7-v6csp5234679imd; Tue, 30 Oct 2018 14:11:11 -0700 (PDT) X-Google-Smtp-Source: AJdET5cinWUwOOU+JSW6+GwfQZIcttvBhMspuPw2HIQpwiiLDw66O5UFqWX2M9Vnmnpa9biRkLEp X-Received: by 2002:a62:7107:: with SMTP id m7-v6mr365949pfc.56.1540933871294; Tue, 30 Oct 2018 14:11:11 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1540933871; cv=none; d=google.com; s=arc-20160816; b=qKquaRrgzbL/oEjKQ9zhtYS1YLYT6e+F41fA67tre9ykeS7g2yTll6FAY3NGwu0MsI g7fzclw8zQ/a2BSJmr16NfEenRRFnaFEEHTlKhFhOkXkVos1cr6Yp/13kVFFwMQ8kWKC d+EhfM0H3VRr0jZSg2LBmQlGvrZaQuF3Y01RhmDuTrYCxeBpQLxraDhJ4cp1AVbRNVEr zqfdqie+SoGx9ngF0RSi+XGQVdmH+Xo79dgUSfp+ckfowasqDXFty7seMGF4bU9eW6Qo ubvCMwTu8xd59uMxtM3m5xkOtwb50LnAjXst4r8eaSf7A5hC2I2MSK3LfKx7U7kA2onU RYfQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:from; bh=tHteMkwIR55yiOGIVdI9/sYgMNz/XjVAj8Zl+QWM9c4=; b=HiZD0+LJi0UquwvgcNWblgNsgUv0dhs/tg8q/DKWc2EjNcJNF4sMR/giQYKJ0msj/b K0EEVwXd1C/r6DfYfRI9qPE8evTWU6eepadGorVeJ0rYNnVZLmf9YjIUbimzgDPW+fr+ umqmFvlhmFRQS3o+0cP1V3jV6/LEsO1JqGavd6mHlpBJFfZ7j1y6i2p4XMw62b+5JAQ2 7uiJlHPF0rxfgT6nztYwKZICrGGmJPjl5pkyeJ0bLCAF4pJzeiq4VzSUYsbBGOMYaRDT zXTt9N5cs00n+cYP9fIjxoDxoDBnT60dHOCyTQb7DHRvrAhYJy4eDwhiXUtH3S5L/6Rz CjXg== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id k27-v6si11695049pgi.549.2018.10.30.14.10.56; Tue, 30 Oct 2018 14:11:11 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728140AbeJaGEi (ORCPT + 99 others); Wed, 31 Oct 2018 02:04:38 -0400 Received: from mga09.intel.com ([134.134.136.24]:28959 "EHLO mga09.intel.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727453AbeJaGEh (ORCPT ); Wed, 31 Oct 2018 02:04:37 -0400 X-Amp-Result: SKIPPED(no attachment in message) X-Amp-File-Uploaded: False Received: from fmsmga002.fm.intel.com ([10.253.24.26]) by orsmga102.jf.intel.com with ESMTP/TLS/DHE-RSA-AES256-GCM-SHA384; 30 Oct 2018 14:09:32 -0700 X-ExtLoop1: 1 X-IronPort-AV: E=Sophos;i="5.54,446,1534834800"; d="scan'208";a="100110068" Received: from maru.jf.intel.com ([10.54.51.77]) by fmsmga002.fm.intel.com with ESMTP; 30 Oct 2018 14:09:31 -0700 From: Jae Hyun Yoo To: Wolfram Sang , Brendan Higgins , Rob Herring , Joel Stanley , Benjamin Herrenschmidt , Mark Rutland , Andrew Jeffery , linux-i2c@vger.kernel.org, openbmc@lists.ozlabs.org, devicetree@vger.kernel.org, linux-arm-kernel@lists.infradead.org, linux-aspeed@lists.ozlabs.org, linux-kernel@vger.kernel.org Cc: Jarkko Nikula , James Feist , Vernon Mauery , Jae Hyun Yoo Subject: [PATCH i2c-next v9 5/5] i2c: aspeed: Add bus idle waiting logic for multi-master use cases Date: Tue, 30 Oct 2018 14:09:16 -0700 Message-Id: <20181030210917.32711-6-jae.hyun.yoo@linux.intel.com> X-Mailer: git-send-email 2.19.1 In-Reply-To: <20181030210917.32711-1-jae.hyun.yoo@linux.intel.com> References: <20181030210917.32711-1-jae.hyun.yoo@linux.intel.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org In multi-master environment, this driver's master cannot know exactly when a peer master sends data to this driver's slave so a case can be happened that this master tries to send data through the master_xfer function but slave data from peer master is still being processed by this driver. To prevent any state corruption in the case, this patch adds checking code if any slave operation is ongoing and it waits up to the bus timeout duration before starting a master_xfer operation. Signed-off-by: Jae Hyun Yoo Reviewed-by: Brendan Higgins --- drivers/i2c/busses/i2c-aspeed.c | 55 ++++++++++++++++++++++++--------- 1 file changed, 40 insertions(+), 15 deletions(-) diff --git a/drivers/i2c/busses/i2c-aspeed.c b/drivers/i2c/busses/i2c-aspeed.c index 833b6b6a4c7e..30c3ab3a4844 100644 --- a/drivers/i2c/busses/i2c-aspeed.c +++ b/drivers/i2c/busses/i2c-aspeed.c @@ -12,6 +12,7 @@ #include #include +#include #include #include #include @@ -115,6 +116,9 @@ /* 0x18 : I2CD Slave Device Address Register */ #define ASPEED_I2CD_DEV_ADDR_MASK GENMASK(6, 0) +/* Busy checking */ +#define ASPEED_I2C_BUS_BUSY_CHECK_INTERVAL_US (10 * 1000) + enum aspeed_i2c_master_state { ASPEED_I2C_MASTER_INACTIVE, ASPEED_I2C_MASTER_START, @@ -156,6 +160,8 @@ struct aspeed_i2c_bus { int cmd_err; /* Protected only by i2c_lock_bus */ int master_xfer_result; + /* Multi-master */ + bool multi_master; #if IS_ENABLED(CONFIG_I2C_SLAVE) struct i2c_client *slave; enum aspeed_i2c_slave_state slave_state; @@ -596,27 +602,44 @@ static irqreturn_t aspeed_i2c_bus_irq(int irq, void *dev_id) return irq_remaining ? IRQ_NONE : IRQ_HANDLED; } +static int aspeed_i2c_check_bus_busy(struct aspeed_i2c_bus *bus) +{ + unsigned long timeout; + + if (bus->multi_master) { + might_sleep(); + /* Initialize it only when multi_master is set */ + timeout = jiffies + bus->adap.timeout; + } + + for (;;) { + if (!(readl(bus->base + ASPEED_I2C_CMD_REG) & + ASPEED_I2CD_BUS_BUSY_STS)) +#if IS_ENABLED(CONFIG_I2C_SLAVE) + if (bus->slave_state == ASPEED_I2C_SLAVE_STOP) +#endif + return 0; + if (!bus->multi_master) + break; + if (time_after(jiffies, timeout)) + break; + usleep_range((ASPEED_I2C_BUS_BUSY_CHECK_INTERVAL_US >> 2) + 1, + ASPEED_I2C_BUS_BUSY_CHECK_INTERVAL_US); + } + + return aspeed_i2c_recover_bus(bus); +} + static int aspeed_i2c_master_xfer(struct i2c_adapter *adap, struct i2c_msg *msgs, int num) { struct aspeed_i2c_bus *bus = i2c_get_adapdata(adap); unsigned long time_left, flags; - int ret = 0; - spin_lock_irqsave(&bus->lock, flags); - bus->cmd_err = 0; - - /* If bus is busy, attempt recovery. We assume a single master - * environment. - */ - if (readl(bus->base + ASPEED_I2C_CMD_REG) & ASPEED_I2CD_BUS_BUSY_STS) { - spin_unlock_irqrestore(&bus->lock, flags); - ret = aspeed_i2c_recover_bus(bus); - if (ret) - return ret; - spin_lock_irqsave(&bus->lock, flags); - } + if (aspeed_i2c_check_bus_busy(bus)) + return -EAGAIN; + spin_lock_irqsave(&bus->lock, flags); bus->cmd_err = 0; bus->msgs = msgs; bus->msgs_index = 0; @@ -827,7 +850,9 @@ static int aspeed_i2c_init(struct aspeed_i2c_bus *bus, if (ret < 0) return ret; - if (!of_property_read_bool(pdev->dev.of_node, "multi-master")) + if (of_property_read_bool(pdev->dev.of_node, "multi-master")) + bus->multi_master = true; + else fun_ctrl_reg |= ASPEED_I2CD_MULTI_MASTER_DIS; /* Enable Master Mode */ -- 2.19.1