aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlexey Starikovskiy <astarikovskiy@suse.de>2008-11-08 13:42:30 -0500
committerLen Brown <len.brown@intel.com>2008-11-11 18:34:19 -0500
commitdd15f8c42af09031e27da5b4d697ce925511f2e1 (patch)
tree284d25e1c8387df3672d88b9108c06b283b6bad5
parentf8248434e6a11d7cd314281be3b39bbcf82fc243 (diff)
ACPI: EC: wait for last write gpe
There is a possibility that EC might break if next command is issued within 1 us after write or burst-disable command. Suggestd-by: Zhao Yakui <yakui.zhao@intel.com> Signed-off-by: Alexey Starikovskiy <astarikovskiy@suse.de> Signed-off-by: Len Brown <len.brown@intel.com>
-rw-r--r--drivers/acpi/ec.c21
1 files changed, 13 insertions, 8 deletions
diff --git a/drivers/acpi/ec.c b/drivers/acpi/ec.c
index b340e08cf1d9..cebd65d2e2a9 100644
--- a/drivers/acpi/ec.c
+++ b/drivers/acpi/ec.c
@@ -102,6 +102,7 @@ struct transaction {
102 u8 command; 102 u8 command;
103 u8 wlen; 103 u8 wlen;
104 u8 rlen; 104 u8 rlen;
105 bool done;
105}; 106};
106 107
107static struct acpi_ec { 108static struct acpi_ec {
@@ -178,7 +179,7 @@ static int ec_transaction_done(struct acpi_ec *ec)
178 unsigned long flags; 179 unsigned long flags;
179 int ret = 0; 180 int ret = 0;
180 spin_lock_irqsave(&ec->curr_lock, flags); 181 spin_lock_irqsave(&ec->curr_lock, flags);
181 if (!ec->curr || (!ec->curr->wlen && !ec->curr->rlen)) 182 if (!ec->curr || ec->curr->done)
182 ret = 1; 183 ret = 1;
183 spin_unlock_irqrestore(&ec->curr_lock, flags); 184 spin_unlock_irqrestore(&ec->curr_lock, flags);
184 return ret; 185 return ret;
@@ -195,17 +196,20 @@ static void gpe_transaction(struct acpi_ec *ec, u8 status)
195 acpi_ec_write_data(ec, *(ec->curr->wdata++)); 196 acpi_ec_write_data(ec, *(ec->curr->wdata++));
196 --ec->curr->wlen; 197 --ec->curr->wlen;
197 } else 198 } else
198 /* false interrupt, state didn't change */ 199 goto err;
199 ++ec->curr->irq_count;
200
201 } else if (ec->curr->rlen > 0) { 200 } else if (ec->curr->rlen > 0) {
202 if ((status & ACPI_EC_FLAG_OBF) == 1) { 201 if ((status & ACPI_EC_FLAG_OBF) == 1) {
203 *(ec->curr->rdata++) = acpi_ec_read_data(ec); 202 *(ec->curr->rdata++) = acpi_ec_read_data(ec);
204 --ec->curr->rlen; 203 if (--ec->curr->rlen == 0)
204 ec->curr->done = true;
205 } else 205 } else
206 /* false interrupt, state didn't change */ 206 goto err;
207 ++ec->curr->irq_count; 207 } else if (ec->curr->wlen == 0 && (status & ACPI_EC_FLAG_IBF) == 0)
208 } 208 ec->curr->done = true;
209 goto unlock;
210err:
211 /* false interrupt, state didn't change */
212 ++ec->curr->irq_count;
209unlock: 213unlock:
210 spin_unlock_irqrestore(&ec->curr_lock, flags); 214 spin_unlock_irqrestore(&ec->curr_lock, flags);
211} 215}
@@ -265,6 +269,7 @@ static int acpi_ec_transaction_unlocked(struct acpi_ec *ec,
265 spin_lock_irqsave(&ec->curr_lock, tmp); 269 spin_lock_irqsave(&ec->curr_lock, tmp);
266 /* following two actions should be kept atomic */ 270 /* following two actions should be kept atomic */
267 t->irq_count = 0; 271 t->irq_count = 0;
272 t->done = false;
268 ec->curr = t; 273 ec->curr = t;
269 acpi_ec_write_cmd(ec, ec->curr->command); 274 acpi_ec_write_cmd(ec, ec->curr->command);
270 if (ec->curr->command == ACPI_EC_COMMAND_QUERY) 275 if (ec->curr->command == ACPI_EC_COMMAND_QUERY)