diff options
| author | Jean-François Dagenais <dagenaisj@sonatest.com> | 2011-05-26 19:26:02 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2011-05-26 20:12:38 -0400 |
| commit | 26a6afb917a8e3eb603358be1238a69e8a16d0ee (patch) | |
| tree | 54632a0485849b0295af922aeeabe19175615edf /drivers/w1 | |
| parent | 89610274bd43edc68c66ff7cf58e05debd519a5e (diff) | |
w1: complete the 1-wire (w1) ds1wm driver search algorithm
This adds multi-slave support of the w1 bus for the ds1wm Synthesizable
1-Wire Bus Master. Also many fixes and tweaks based on the rev3 of the
datasheet http://datasheets.maxim-ic.com/en/ds/DS1WM.pdf
Signed-off-by: Jean-François Dagenais <dagenaisj@sonatest.com>
Cc: Evgeniy Polyakov <johnpol@2ka.mipt.ru>
Cc: Szabolcs Gyurko <szabolcs.gyurko@tlt.hu>
Cc: Matt Reimer <mreimer@vpop.net>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'drivers/w1')
| -rw-r--r-- | drivers/w1/masters/Kconfig | 2 | ||||
| -rw-r--r-- | drivers/w1/masters/ds1wm.c | 321 |
2 files changed, 220 insertions, 103 deletions
diff --git a/drivers/w1/masters/Kconfig b/drivers/w1/masters/Kconfig index 7c608c5ccf84..00d615d7aa21 100644 --- a/drivers/w1/masters/Kconfig +++ b/drivers/w1/masters/Kconfig | |||
| @@ -42,7 +42,7 @@ config W1_MASTER_MXC | |||
| 42 | 42 | ||
| 43 | config W1_MASTER_DS1WM | 43 | config W1_MASTER_DS1WM |
| 44 | tristate "Maxim DS1WM 1-wire busmaster" | 44 | tristate "Maxim DS1WM 1-wire busmaster" |
| 45 | depends on W1 && ARM && HAVE_CLK | 45 | depends on W1 |
| 46 | help | 46 | help |
| 47 | Say Y here to enable the DS1WM 1-wire driver, such as that | 47 | Say Y here to enable the DS1WM 1-wire driver, such as that |
| 48 | in HP iPAQ devices like h5xxx, h2200, and ASIC3-based like | 48 | in HP iPAQ devices like h5xxx, h2200, and ASIC3-based like |
diff --git a/drivers/w1/masters/ds1wm.c b/drivers/w1/masters/ds1wm.c index 0855d6cce3c1..ad57593d224a 100644 --- a/drivers/w1/masters/ds1wm.c +++ b/drivers/w1/masters/ds1wm.c | |||
| @@ -33,6 +33,7 @@ | |||
| 33 | #define DS1WM_INT 0x02 /* R/W interrupt status */ | 33 | #define DS1WM_INT 0x02 /* R/W interrupt status */ |
| 34 | #define DS1WM_INT_EN 0x03 /* R/W interrupt enable */ | 34 | #define DS1WM_INT_EN 0x03 /* R/W interrupt enable */ |
| 35 | #define DS1WM_CLKDIV 0x04 /* R/W 5 bits of divisor and pre-scale */ | 35 | #define DS1WM_CLKDIV 0x04 /* R/W 5 bits of divisor and pre-scale */ |
| 36 | #define DS1WM_CNTRL 0x05 /* R/W master control register (not used yet) */ | ||
| 36 | 37 | ||
| 37 | #define DS1WM_CMD_1W_RESET (1 << 0) /* force reset on 1-wire bus */ | 38 | #define DS1WM_CMD_1W_RESET (1 << 0) /* force reset on 1-wire bus */ |
| 38 | #define DS1WM_CMD_SRA (1 << 1) /* enable Search ROM accelerator mode */ | 39 | #define DS1WM_CMD_SRA (1 << 1) /* enable Search ROM accelerator mode */ |
| @@ -56,6 +57,7 @@ | |||
| 56 | #define DS1WM_INTEN_ERSRF (1 << 5) /* enable rx shift register full int */ | 57 | #define DS1WM_INTEN_ERSRF (1 << 5) /* enable rx shift register full int */ |
| 57 | #define DS1WM_INTEN_DQO (1 << 6) /* enable direct bus driving ops */ | 58 | #define DS1WM_INTEN_DQO (1 << 6) /* enable direct bus driving ops */ |
| 58 | 59 | ||
| 60 | #define DS1WM_INTEN_NOT_IAS (~DS1WM_INTEN_IAS) /* all but INTR active state */ | ||
| 59 | 61 | ||
| 60 | #define DS1WM_TIMEOUT (HZ * 5) | 62 | #define DS1WM_TIMEOUT (HZ * 5) |
| 61 | 63 | ||
| @@ -63,41 +65,50 @@ static struct { | |||
| 63 | unsigned long freq; | 65 | unsigned long freq; |
| 64 | unsigned long divisor; | 66 | unsigned long divisor; |
| 65 | } freq[] = { | 67 | } freq[] = { |
| 66 | { 4000000, 0x8 }, | 68 | { 1000000, 0x80 }, |
| 67 | { 5000000, 0x2 }, | 69 | { 2000000, 0x84 }, |
| 68 | { 6000000, 0x5 }, | 70 | { 3000000, 0x81 }, |
| 69 | { 7000000, 0x3 }, | 71 | { 4000000, 0x88 }, |
| 70 | { 8000000, 0xc }, | 72 | { 5000000, 0x82 }, |
| 71 | { 10000000, 0x6 }, | 73 | { 6000000, 0x85 }, |
| 72 | { 12000000, 0x9 }, | 74 | { 7000000, 0x83 }, |
| 73 | { 14000000, 0x7 }, | 75 | { 8000000, 0x8c }, |
| 74 | { 16000000, 0x10 }, | 76 | { 10000000, 0x86 }, |
| 75 | { 20000000, 0xa }, | 77 | { 12000000, 0x89 }, |
| 76 | { 24000000, 0xd }, | 78 | { 14000000, 0x87 }, |
| 77 | { 28000000, 0xb }, | 79 | { 16000000, 0x90 }, |
| 78 | { 32000000, 0x14 }, | 80 | { 20000000, 0x8a }, |
| 79 | { 40000000, 0xe }, | 81 | { 24000000, 0x8d }, |
| 80 | { 48000000, 0x11 }, | 82 | { 28000000, 0x8b }, |
| 81 | { 56000000, 0xf }, | 83 | { 32000000, 0x94 }, |
| 82 | { 64000000, 0x18 }, | 84 | { 40000000, 0x8e }, |
| 83 | { 80000000, 0x12 }, | 85 | { 48000000, 0x91 }, |
| 84 | { 96000000, 0x15 }, | 86 | { 56000000, 0x8f }, |
| 85 | { 112000000, 0x13 }, | 87 | { 64000000, 0x98 }, |
| 86 | { 128000000, 0x1c }, | 88 | { 80000000, 0x92 }, |
| 89 | { 96000000, 0x95 }, | ||
| 90 | { 112000000, 0x93 }, | ||
| 91 | { 128000000, 0x9c }, | ||
| 92 | /* you can continue this table, consult the OPERATION - CLOCK DIVISOR | ||
| 93 | section of the ds1wm spec sheet. */ | ||
| 87 | }; | 94 | }; |
| 88 | 95 | ||
| 89 | struct ds1wm_data { | 96 | struct ds1wm_data { |
| 90 | void __iomem *map; | 97 | void __iomem *map; |
| 91 | int bus_shift; /* # of shifts to calc register offsets */ | 98 | int bus_shift; /* # of shifts to calc register offsets */ |
| 92 | struct platform_device *pdev; | 99 | struct platform_device *pdev; |
| 93 | const struct mfd_cell *cell; | 100 | const struct mfd_cell *cell; |
| 94 | int irq; | 101 | int irq; |
| 95 | int active_high; | 102 | int slave_present; |
| 96 | int slave_present; | 103 | void *reset_complete; |
| 97 | void *reset_complete; | 104 | void *read_complete; |
| 98 | void *read_complete; | 105 | void *write_complete; |
| 99 | void *write_complete; | 106 | int read_error; |
| 100 | u8 read_byte; /* last byte received */ | 107 | /* last byte received */ |
| 108 | u8 read_byte; | ||
| 109 | /* byte to write that makes all intr disabled, */ | ||
| 110 | /* considering active_state (IAS) (optimization) */ | ||
| 111 | u8 int_en_reg_none; | ||
| 101 | }; | 112 | }; |
| 102 | 113 | ||
| 103 | static inline void ds1wm_write_register(struct ds1wm_data *ds1wm_data, u32 reg, | 114 | static inline void ds1wm_write_register(struct ds1wm_data *ds1wm_data, u32 reg, |
| @@ -115,23 +126,39 @@ static inline u8 ds1wm_read_register(struct ds1wm_data *ds1wm_data, u32 reg) | |||
| 115 | static irqreturn_t ds1wm_isr(int isr, void *data) | 126 | static irqreturn_t ds1wm_isr(int isr, void *data) |
| 116 | { | 127 | { |
| 117 | struct ds1wm_data *ds1wm_data = data; | 128 | struct ds1wm_data *ds1wm_data = data; |
| 118 | u8 intr = ds1wm_read_register(ds1wm_data, DS1WM_INT); | 129 | u8 intr; |
| 130 | u8 inten = ds1wm_read_register(ds1wm_data, DS1WM_INT_EN); | ||
| 131 | /* if no bits are set in int enable register (except the IAS) | ||
| 132 | than go no further, reading the regs below has side effects */ | ||
| 133 | if (!(inten & DS1WM_INTEN_NOT_IAS)) | ||
| 134 | return IRQ_NONE; | ||
| 119 | 135 | ||
| 120 | ds1wm_data->slave_present = (intr & DS1WM_INT_PDR) ? 0 : 1; | 136 | ds1wm_write_register(ds1wm_data, |
| 137 | DS1WM_INT_EN, ds1wm_data->int_en_reg_none); | ||
| 121 | 138 | ||
| 122 | if ((intr & DS1WM_INT_PD) && ds1wm_data->reset_complete) | 139 | /* this read action clears the INTR and certain flags in ds1wm */ |
| 123 | complete(ds1wm_data->reset_complete); | 140 | intr = ds1wm_read_register(ds1wm_data, DS1WM_INT); |
| 124 | 141 | ||
| 125 | if ((intr & DS1WM_INT_TSRE) && ds1wm_data->write_complete) | 142 | ds1wm_data->slave_present = (intr & DS1WM_INT_PDR) ? 0 : 1; |
| 126 | complete(ds1wm_data->write_complete); | ||
| 127 | 143 | ||
| 144 | if ((intr & DS1WM_INT_TSRE) && ds1wm_data->write_complete) { | ||
| 145 | inten &= ~DS1WM_INTEN_ETMT; | ||
| 146 | complete(ds1wm_data->write_complete); | ||
| 147 | } | ||
| 128 | if (intr & DS1WM_INT_RBF) { | 148 | if (intr & DS1WM_INT_RBF) { |
| 149 | /* this read clears the RBF flag */ | ||
| 129 | ds1wm_data->read_byte = ds1wm_read_register(ds1wm_data, | 150 | ds1wm_data->read_byte = ds1wm_read_register(ds1wm_data, |
| 130 | DS1WM_DATA); | 151 | DS1WM_DATA); |
| 152 | inten &= ~DS1WM_INTEN_ERBF; | ||
| 131 | if (ds1wm_data->read_complete) | 153 | if (ds1wm_data->read_complete) |
| 132 | complete(ds1wm_data->read_complete); | 154 | complete(ds1wm_data->read_complete); |
| 133 | } | 155 | } |
| 156 | if ((intr & DS1WM_INT_PD) && ds1wm_data->reset_complete) { | ||
| 157 | inten &= ~DS1WM_INTEN_EPD; | ||
| 158 | complete(ds1wm_data->reset_complete); | ||
| 159 | } | ||
| 134 | 160 | ||
| 161 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, inten); | ||
| 135 | return IRQ_HANDLED; | 162 | return IRQ_HANDLED; |
| 136 | } | 163 | } |
| 137 | 164 | ||
| @@ -142,33 +169,19 @@ static int ds1wm_reset(struct ds1wm_data *ds1wm_data) | |||
| 142 | 169 | ||
| 143 | ds1wm_data->reset_complete = &reset_done; | 170 | ds1wm_data->reset_complete = &reset_done; |
| 144 | 171 | ||
| 172 | /* enable Presence detect only */ | ||
| 145 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, DS1WM_INTEN_EPD | | 173 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, DS1WM_INTEN_EPD | |
| 146 | (ds1wm_data->active_high ? DS1WM_INTEN_IAS : 0)); | 174 | ds1wm_data->int_en_reg_none); |
| 147 | 175 | ||
| 148 | ds1wm_write_register(ds1wm_data, DS1WM_CMD, DS1WM_CMD_1W_RESET); | 176 | ds1wm_write_register(ds1wm_data, DS1WM_CMD, DS1WM_CMD_1W_RESET); |
| 149 | 177 | ||
| 150 | timeleft = wait_for_completion_timeout(&reset_done, DS1WM_TIMEOUT); | 178 | timeleft = wait_for_completion_timeout(&reset_done, DS1WM_TIMEOUT); |
| 151 | ds1wm_data->reset_complete = NULL; | 179 | ds1wm_data->reset_complete = NULL; |
| 152 | if (!timeleft) { | 180 | if (!timeleft) { |
| 153 | dev_err(&ds1wm_data->pdev->dev, "reset failed\n"); | 181 | dev_err(&ds1wm_data->pdev->dev, "reset failed, timed out\n"); |
| 154 | return 1; | 182 | return 1; |
| 155 | } | 183 | } |
| 156 | 184 | ||
| 157 | /* Wait for the end of the reset. According to the specs, the time | ||
| 158 | * from when the interrupt is asserted to the end of the reset is: | ||
| 159 | * tRSTH - tPDH - tPDL - tPDI | ||
| 160 | * 625 us - 60 us - 240 us - 100 ns = 324.9 us | ||
| 161 | * | ||
| 162 | * We'll wait a bit longer just to be sure. | ||
| 163 | * Was udelay(500), but if it is going to busywait the cpu that long, | ||
| 164 | * might as well come back later. | ||
| 165 | */ | ||
| 166 | msleep(1); | ||
| 167 | |||
| 168 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, | ||
| 169 | DS1WM_INTEN_ERBF | DS1WM_INTEN_ETMT | DS1WM_INTEN_EPD | | ||
| 170 | (ds1wm_data->active_high ? DS1WM_INTEN_IAS : 0)); | ||
| 171 | |||
| 172 | if (!ds1wm_data->slave_present) { | 185 | if (!ds1wm_data->slave_present) { |
| 173 | dev_dbg(&ds1wm_data->pdev->dev, "reset: no devices found\n"); | 186 | dev_dbg(&ds1wm_data->pdev->dev, "reset: no devices found\n"); |
| 174 | return 1; | 187 | return 1; |
| @@ -179,26 +192,47 @@ static int ds1wm_reset(struct ds1wm_data *ds1wm_data) | |||
| 179 | 192 | ||
| 180 | static int ds1wm_write(struct ds1wm_data *ds1wm_data, u8 data) | 193 | static int ds1wm_write(struct ds1wm_data *ds1wm_data, u8 data) |
| 181 | { | 194 | { |
| 195 | unsigned long timeleft; | ||
| 182 | DECLARE_COMPLETION_ONSTACK(write_done); | 196 | DECLARE_COMPLETION_ONSTACK(write_done); |
| 183 | ds1wm_data->write_complete = &write_done; | 197 | ds1wm_data->write_complete = &write_done; |
| 184 | 198 | ||
| 199 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, | ||
| 200 | ds1wm_data->int_en_reg_none | DS1WM_INTEN_ETMT); | ||
| 201 | |||
| 185 | ds1wm_write_register(ds1wm_data, DS1WM_DATA, data); | 202 | ds1wm_write_register(ds1wm_data, DS1WM_DATA, data); |
| 186 | 203 | ||
| 187 | wait_for_completion_timeout(&write_done, DS1WM_TIMEOUT); | 204 | timeleft = wait_for_completion_timeout(&write_done, DS1WM_TIMEOUT); |
| 205 | |||
| 188 | ds1wm_data->write_complete = NULL; | 206 | ds1wm_data->write_complete = NULL; |
| 207 | if (!timeleft) { | ||
| 208 | dev_err(&ds1wm_data->pdev->dev, "write failed, timed out\n"); | ||
| 209 | return -ETIMEDOUT; | ||
| 210 | } | ||
| 189 | 211 | ||
| 190 | return 0; | 212 | return 0; |
| 191 | } | 213 | } |
| 192 | 214 | ||
| 193 | static int ds1wm_read(struct ds1wm_data *ds1wm_data, unsigned char write_data) | 215 | static u8 ds1wm_read(struct ds1wm_data *ds1wm_data, unsigned char write_data) |
| 194 | { | 216 | { |
| 217 | unsigned long timeleft; | ||
| 218 | u8 intEnable = DS1WM_INTEN_ERBF | ds1wm_data->int_en_reg_none; | ||
| 195 | DECLARE_COMPLETION_ONSTACK(read_done); | 219 | DECLARE_COMPLETION_ONSTACK(read_done); |
| 220 | |||
| 221 | ds1wm_read_register(ds1wm_data, DS1WM_DATA); | ||
| 222 | |||
| 196 | ds1wm_data->read_complete = &read_done; | 223 | ds1wm_data->read_complete = &read_done; |
| 224 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, intEnable); | ||
| 197 | 225 | ||
| 198 | ds1wm_write(ds1wm_data, write_data); | 226 | ds1wm_write_register(ds1wm_data, DS1WM_DATA, write_data); |
| 199 | wait_for_completion_timeout(&read_done, DS1WM_TIMEOUT); | 227 | timeleft = wait_for_completion_timeout(&read_done, DS1WM_TIMEOUT); |
| 200 | ds1wm_data->read_complete = NULL; | ||
| 201 | 228 | ||
| 229 | ds1wm_data->read_complete = NULL; | ||
| 230 | if (!timeleft) { | ||
| 231 | dev_err(&ds1wm_data->pdev->dev, "read failed, timed out\n"); | ||
| 232 | ds1wm_data->read_error = -ETIMEDOUT; | ||
| 233 | return 0xFF; | ||
| 234 | } | ||
| 235 | ds1wm_data->read_error = 0; | ||
| 202 | return ds1wm_data->read_byte; | 236 | return ds1wm_data->read_byte; |
| 203 | } | 237 | } |
| 204 | 238 | ||
| @@ -206,8 +240,8 @@ static int ds1wm_find_divisor(int gclk) | |||
| 206 | { | 240 | { |
| 207 | int i; | 241 | int i; |
| 208 | 242 | ||
| 209 | for (i = 0; i < ARRAY_SIZE(freq); i++) | 243 | for (i = ARRAY_SIZE(freq)-1; i >= 0; --i) |
| 210 | if (gclk <= freq[i].freq) | 244 | if (gclk >= freq[i].freq) |
| 211 | return freq[i].divisor; | 245 | return freq[i].divisor; |
| 212 | 246 | ||
| 213 | return 0; | 247 | return 0; |
| @@ -222,6 +256,8 @@ static void ds1wm_up(struct ds1wm_data *ds1wm_data) | |||
| 222 | ds1wm_data->cell->enable(ds1wm_data->pdev); | 256 | ds1wm_data->cell->enable(ds1wm_data->pdev); |
| 223 | 257 | ||
| 224 | divisor = ds1wm_find_divisor(plat->clock_rate); | 258 | divisor = ds1wm_find_divisor(plat->clock_rate); |
| 259 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 260 | "found divisor 0x%x for clock %d\n", divisor, plat->clock_rate); | ||
| 225 | if (divisor == 0) { | 261 | if (divisor == 0) { |
| 226 | dev_err(&ds1wm_data->pdev->dev, | 262 | dev_err(&ds1wm_data->pdev->dev, |
| 227 | "no suitable divisor for %dHz clock\n", | 263 | "no suitable divisor for %dHz clock\n", |
| @@ -242,7 +278,7 @@ static void ds1wm_down(struct ds1wm_data *ds1wm_data) | |||
| 242 | 278 | ||
| 243 | /* Disable interrupts. */ | 279 | /* Disable interrupts. */ |
| 244 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, | 280 | ds1wm_write_register(ds1wm_data, DS1WM_INT_EN, |
| 245 | ds1wm_data->active_high ? DS1WM_INTEN_IAS : 0); | 281 | ds1wm_data->int_en_reg_none); |
| 246 | 282 | ||
| 247 | if (ds1wm_data->cell->disable) | 283 | if (ds1wm_data->cell->disable) |
| 248 | ds1wm_data->cell->disable(ds1wm_data->pdev); | 284 | ds1wm_data->cell->disable(ds1wm_data->pdev); |
| @@ -279,41 +315,121 @@ static void ds1wm_search(void *data, struct w1_master *master_dev, | |||
| 279 | { | 315 | { |
| 280 | struct ds1wm_data *ds1wm_data = data; | 316 | struct ds1wm_data *ds1wm_data = data; |
| 281 | int i; | 317 | int i; |
| 282 | unsigned long long rom_id; | 318 | int ms_discrep_bit = -1; |
| 283 | 319 | u64 r = 0; /* holds the progress of the search */ | |
| 284 | /* XXX We need to iterate for multiple devices per the DS1WM docs. | 320 | u64 r_prime, d; |
| 285 | * See http://www.maxim-ic.com/appnotes.cfm/appnote_number/120. */ | 321 | unsigned slaves_found = 0; |
| 286 | if (ds1wm_reset(ds1wm_data)) | 322 | unsigned int pass = 0; |
| 287 | return; | 323 | |
| 288 | 324 | dev_dbg(&ds1wm_data->pdev->dev, "search begin\n"); | |
| 289 | ds1wm_write(ds1wm_data, search_type); | 325 | while (true) { |
| 290 | ds1wm_write_register(ds1wm_data, DS1WM_CMD, DS1WM_CMD_SRA); | 326 | ++pass; |
| 291 | 327 | if (pass > 100) { | |
| 292 | for (rom_id = 0, i = 0; i < 16; i++) { | 328 | dev_dbg(&ds1wm_data->pdev->dev, |
| 293 | 329 | "too many attempts (100), search aborted\n"); | |
| 294 | unsigned char resp, r, d; | 330 | return; |
| 295 | 331 | } | |
| 296 | resp = ds1wm_read(ds1wm_data, 0x00); | 332 | |
| 297 | 333 | if (ds1wm_reset(ds1wm_data)) { | |
| 298 | r = ((resp & 0x02) >> 1) | | 334 | dev_dbg(&ds1wm_data->pdev->dev, |
| 299 | ((resp & 0x08) >> 2) | | 335 | "pass: %d reset error (or no slaves)\n", pass); |
| 300 | ((resp & 0x20) >> 3) | | 336 | break; |
| 301 | ((resp & 0x80) >> 4); | 337 | } |
| 302 | 338 | ||
| 303 | d = ((resp & 0x01) >> 0) | | 339 | dev_dbg(&ds1wm_data->pdev->dev, |
| 304 | ((resp & 0x04) >> 1) | | 340 | "pass: %d r : %0#18llx writing SEARCH_ROM\n", pass, r); |
| 305 | ((resp & 0x10) >> 2) | | 341 | ds1wm_write(ds1wm_data, search_type); |
| 306 | ((resp & 0x40) >> 3); | 342 | dev_dbg(&ds1wm_data->pdev->dev, |
| 307 | 343 | "pass: %d entering ASM\n", pass); | |
| 308 | rom_id |= (unsigned long long) r << (i * 4); | 344 | ds1wm_write_register(ds1wm_data, DS1WM_CMD, DS1WM_CMD_SRA); |
| 309 | 345 | dev_dbg(&ds1wm_data->pdev->dev, | |
| 310 | } | 346 | "pass: %d begining nibble loop\n", pass); |
| 311 | dev_dbg(&ds1wm_data->pdev->dev, "found 0x%08llX\n", rom_id); | 347 | |
| 312 | 348 | r_prime = 0; | |
| 313 | ds1wm_write_register(ds1wm_data, DS1WM_CMD, ~DS1WM_CMD_SRA); | 349 | d = 0; |
| 314 | ds1wm_reset(ds1wm_data); | 350 | /* we work one nibble at a time */ |
| 315 | 351 | /* each nibble is interleaved to form a byte */ | |
| 316 | slave_found(master_dev, rom_id); | 352 | for (i = 0; i < 16; i++) { |
| 353 | |||
| 354 | unsigned char resp, _r, _r_prime, _d; | ||
| 355 | |||
| 356 | _r = (r >> (4*i)) & 0xf; | ||
| 357 | _r = ((_r & 0x1) << 1) | | ||
| 358 | ((_r & 0x2) << 2) | | ||
| 359 | ((_r & 0x4) << 3) | | ||
| 360 | ((_r & 0x8) << 4); | ||
| 361 | |||
| 362 | /* writes _r, then reads back: */ | ||
| 363 | resp = ds1wm_read(ds1wm_data, _r); | ||
| 364 | |||
| 365 | if (ds1wm_data->read_error) { | ||
| 366 | dev_err(&ds1wm_data->pdev->dev, | ||
| 367 | "pass: %d nibble: %d read error\n", pass, i); | ||
| 368 | break; | ||
| 369 | } | ||
| 370 | |||
| 371 | _r_prime = ((resp & 0x02) >> 1) | | ||
| 372 | ((resp & 0x08) >> 2) | | ||
| 373 | ((resp & 0x20) >> 3) | | ||
| 374 | ((resp & 0x80) >> 4); | ||
| 375 | |||
| 376 | _d = ((resp & 0x01) >> 0) | | ||
| 377 | ((resp & 0x04) >> 1) | | ||
| 378 | ((resp & 0x10) >> 2) | | ||
| 379 | ((resp & 0x40) >> 3); | ||
| 380 | |||
| 381 | r_prime |= (unsigned long long) _r_prime << (i * 4); | ||
| 382 | d |= (unsigned long long) _d << (i * 4); | ||
| 383 | |||
| 384 | } | ||
| 385 | if (ds1wm_data->read_error) { | ||
| 386 | dev_err(&ds1wm_data->pdev->dev, | ||
| 387 | "pass: %d read error, retrying\n", pass); | ||
| 388 | break; | ||
| 389 | } | ||
| 390 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 391 | "pass: %d r\': %0#18llx d:%0#18llx\n", | ||
| 392 | pass, r_prime, d); | ||
| 393 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 394 | "pass: %d nibble loop complete, exiting ASM\n", pass); | ||
| 395 | ds1wm_write_register(ds1wm_data, DS1WM_CMD, ~DS1WM_CMD_SRA); | ||
| 396 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 397 | "pass: %d resetting bus\n", pass); | ||
| 398 | ds1wm_reset(ds1wm_data); | ||
| 399 | if ((r_prime & ((u64)1 << 63)) && (d & ((u64)1 << 63))) { | ||
| 400 | dev_err(&ds1wm_data->pdev->dev, | ||
| 401 | "pass: %d bus error, retrying\n", pass); | ||
| 402 | continue; /* start over */ | ||
| 403 | } | ||
| 404 | |||
| 405 | |||
| 406 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 407 | "pass: %d found %0#18llx\n", pass, r_prime); | ||
| 408 | slave_found(master_dev, r_prime); | ||
| 409 | ++slaves_found; | ||
| 410 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 411 | "pass: %d complete, preparing next pass\n", pass); | ||
| 412 | |||
| 413 | /* any discrepency found which we already choose the | ||
| 414 | '1' branch is now is now irrelevant we reveal the | ||
| 415 | next branch with this: */ | ||
| 416 | d &= ~r; | ||
| 417 | /* find last bit set, i.e. the most signif. bit set */ | ||
| 418 | ms_discrep_bit = fls64(d) - 1; | ||
| 419 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 420 | "pass: %d new d:%0#18llx MS discrep bit:%d\n", | ||
| 421 | pass, d, ms_discrep_bit); | ||
| 422 | |||
| 423 | /* prev_ms_discrep_bit = ms_discrep_bit; | ||
| 424 | prepare for next ROM search: */ | ||
| 425 | if (ms_discrep_bit == -1) | ||
| 426 | break; | ||
| 427 | |||
| 428 | r = (r & ~(~0ull << (ms_discrep_bit))) | 1 << ms_discrep_bit; | ||
| 429 | } /* end while true */ | ||
| 430 | dev_dbg(&ds1wm_data->pdev->dev, | ||
| 431 | "pass: %d total: %d search done ms d bit pos: %d\n", pass, | ||
| 432 | slaves_found, ms_discrep_bit); | ||
| 317 | } | 433 | } |
| 318 | 434 | ||
| 319 | /* --------------------------------------------------------------------- */ | 435 | /* --------------------------------------------------------------------- */ |
| @@ -373,15 +489,15 @@ static int ds1wm_probe(struct platform_device *pdev) | |||
| 373 | goto err1; | 489 | goto err1; |
| 374 | } | 490 | } |
| 375 | ds1wm_data->irq = res->start; | 491 | ds1wm_data->irq = res->start; |
| 376 | ds1wm_data->active_high = plat->active_high; | 492 | ds1wm_data->int_en_reg_none = (plat->active_high ? DS1WM_INTEN_IAS : 0); |
| 377 | 493 | ||
| 378 | if (res->flags & IORESOURCE_IRQ_HIGHEDGE) | 494 | if (res->flags & IORESOURCE_IRQ_HIGHEDGE) |
| 379 | irq_set_irq_type(ds1wm_data->irq, IRQ_TYPE_EDGE_RISING); | 495 | irq_set_irq_type(ds1wm_data->irq, IRQ_TYPE_EDGE_RISING); |
| 380 | if (res->flags & IORESOURCE_IRQ_LOWEDGE) | 496 | if (res->flags & IORESOURCE_IRQ_LOWEDGE) |
| 381 | irq_set_irq_type(ds1wm_data->irq, IRQ_TYPE_EDGE_FALLING); | 497 | irq_set_irq_type(ds1wm_data->irq, IRQ_TYPE_EDGE_FALLING); |
| 382 | 498 | ||
| 383 | ret = request_irq(ds1wm_data->irq, ds1wm_isr, IRQF_DISABLED, | 499 | ret = request_irq(ds1wm_data->irq, ds1wm_isr, |
| 384 | "ds1wm", ds1wm_data); | 500 | IRQF_DISABLED | IRQF_SHARED, "ds1wm", ds1wm_data); |
| 385 | if (ret) | 501 | if (ret) |
| 386 | goto err1; | 502 | goto err1; |
| 387 | 503 | ||
| @@ -468,5 +584,6 @@ module_exit(ds1wm_exit); | |||
| 468 | 584 | ||
| 469 | MODULE_LICENSE("GPL"); | 585 | MODULE_LICENSE("GPL"); |
| 470 | MODULE_AUTHOR("Szabolcs Gyurko <szabolcs.gyurko@tlt.hu>, " | 586 | MODULE_AUTHOR("Szabolcs Gyurko <szabolcs.gyurko@tlt.hu>, " |
| 471 | "Matt Reimer <mreimer@vpop.net>"); | 587 | "Matt Reimer <mreimer@vpop.net>," |
| 588 | "Jean-Francois Dagenais <dagenaisj@sonatest.com>"); | ||
| 472 | MODULE_DESCRIPTION("DS1WM w1 busmaster driver"); | 589 | MODULE_DESCRIPTION("DS1WM w1 busmaster driver"); |
