diff options
255 files changed, 4602 insertions, 2215 deletions
diff --git a/Documentation/DocBook/media-entities.tmpl b/Documentation/DocBook/media-entities.tmpl index 5d259c632cdf..fea63b45471a 100644 --- a/Documentation/DocBook/media-entities.tmpl +++ b/Documentation/DocBook/media-entities.tmpl | |||
| @@ -294,6 +294,7 @@ | |||
| 294 | <!ENTITY sub-srggb10 SYSTEM "v4l/pixfmt-srggb10.xml"> | 294 | <!ENTITY sub-srggb10 SYSTEM "v4l/pixfmt-srggb10.xml"> |
| 295 | <!ENTITY sub-srggb8 SYSTEM "v4l/pixfmt-srggb8.xml"> | 295 | <!ENTITY sub-srggb8 SYSTEM "v4l/pixfmt-srggb8.xml"> |
| 296 | <!ENTITY sub-y10 SYSTEM "v4l/pixfmt-y10.xml"> | 296 | <!ENTITY sub-y10 SYSTEM "v4l/pixfmt-y10.xml"> |
| 297 | <!ENTITY sub-y12 SYSTEM "v4l/pixfmt-y12.xml"> | ||
| 297 | <!ENTITY sub-pixfmt SYSTEM "v4l/pixfmt.xml"> | 298 | <!ENTITY sub-pixfmt SYSTEM "v4l/pixfmt.xml"> |
| 298 | <!ENTITY sub-cropcap SYSTEM "v4l/vidioc-cropcap.xml"> | 299 | <!ENTITY sub-cropcap SYSTEM "v4l/vidioc-cropcap.xml"> |
| 299 | <!ENTITY sub-dbg-g-register SYSTEM "v4l/vidioc-dbg-g-register.xml"> | 300 | <!ENTITY sub-dbg-g-register SYSTEM "v4l/vidioc-dbg-g-register.xml"> |
diff --git a/Documentation/DocBook/v4l/media-ioc-setup-link.xml b/Documentation/DocBook/v4l/media-ioc-setup-link.xml index 2331e76ded17..cec97af4dab4 100644 --- a/Documentation/DocBook/v4l/media-ioc-setup-link.xml +++ b/Documentation/DocBook/v4l/media-ioc-setup-link.xml | |||
| @@ -34,7 +34,7 @@ | |||
| 34 | <varlistentry> | 34 | <varlistentry> |
| 35 | <term><parameter>request</parameter></term> | 35 | <term><parameter>request</parameter></term> |
| 36 | <listitem> | 36 | <listitem> |
| 37 | <para>MEDIA_IOC_ENUM_LINKS</para> | 37 | <para>MEDIA_IOC_SETUP_LINK</para> |
| 38 | </listitem> | 38 | </listitem> |
| 39 | </varlistentry> | 39 | </varlistentry> |
| 40 | <varlistentry> | 40 | <varlistentry> |
diff --git a/Documentation/DocBook/v4l/pixfmt-y12.xml b/Documentation/DocBook/v4l/pixfmt-y12.xml new file mode 100644 index 000000000000..ff417b858cc9 --- /dev/null +++ b/Documentation/DocBook/v4l/pixfmt-y12.xml | |||
| @@ -0,0 +1,79 @@ | |||
| 1 | <refentry id="V4L2-PIX-FMT-Y12"> | ||
| 2 | <refmeta> | ||
| 3 | <refentrytitle>V4L2_PIX_FMT_Y12 ('Y12 ')</refentrytitle> | ||
| 4 | &manvol; | ||
| 5 | </refmeta> | ||
| 6 | <refnamediv> | ||
| 7 | <refname><constant>V4L2_PIX_FMT_Y12</constant></refname> | ||
| 8 | <refpurpose>Grey-scale image</refpurpose> | ||
| 9 | </refnamediv> | ||
| 10 | <refsect1> | ||
| 11 | <title>Description</title> | ||
| 12 | |||
| 13 | <para>This is a grey-scale image with a depth of 12 bits per pixel. Pixels | ||
| 14 | are stored in 16-bit words with unused high bits padded with 0. The least | ||
| 15 | significant byte is stored at lower memory addresses (little-endian).</para> | ||
| 16 | |||
| 17 | <example> | ||
| 18 | <title><constant>V4L2_PIX_FMT_Y12</constant> 4 × 4 | ||
| 19 | pixel image</title> | ||
| 20 | |||
| 21 | <formalpara> | ||
| 22 | <title>Byte Order.</title> | ||
| 23 | <para>Each cell is one byte. | ||
| 24 | <informaltable frame="none"> | ||
| 25 | <tgroup cols="9" align="center"> | ||
| 26 | <colspec align="left" colwidth="2*" /> | ||
| 27 | <tbody valign="top"> | ||
| 28 | <row> | ||
| 29 | <entry>start + 0:</entry> | ||
| 30 | <entry>Y'<subscript>00low</subscript></entry> | ||
| 31 | <entry>Y'<subscript>00high</subscript></entry> | ||
| 32 | <entry>Y'<subscript>01low</subscript></entry> | ||
| 33 | <entry>Y'<subscript>01high</subscript></entry> | ||
| 34 | <entry>Y'<subscript>02low</subscript></entry> | ||
| 35 | <entry>Y'<subscript>02high</subscript></entry> | ||
| 36 | <entry>Y'<subscript>03low</subscript></entry> | ||
| 37 | <entry>Y'<subscript>03high</subscript></entry> | ||
| 38 | </row> | ||
| 39 | <row> | ||
| 40 | <entry>start + 8:</entry> | ||
| 41 | <entry>Y'<subscript>10low</subscript></entry> | ||
| 42 | <entry>Y'<subscript>10high</subscript></entry> | ||
| 43 | <entry>Y'<subscript>11low</subscript></entry> | ||
| 44 | <entry>Y'<subscript>11high</subscript></entry> | ||
| 45 | <entry>Y'<subscript>12low</subscript></entry> | ||
| 46 | <entry>Y'<subscript>12high</subscript></entry> | ||
| 47 | <entry>Y'<subscript>13low</subscript></entry> | ||
| 48 | <entry>Y'<subscript>13high</subscript></entry> | ||
| 49 | </row> | ||
| 50 | <row> | ||
| 51 | <entry>start + 16:</entry> | ||
| 52 | <entry>Y'<subscript>20low</subscript></entry> | ||
| 53 | <entry>Y'<subscript>20high</subscript></entry> | ||
| 54 | <entry>Y'<subscript>21low</subscript></entry> | ||
| 55 | <entry>Y'<subscript>21high</subscript></entry> | ||
| 56 | <entry>Y'<subscript>22low</subscript></entry> | ||
| 57 | <entry>Y'<subscript>22high</subscript></entry> | ||
| 58 | <entry>Y'<subscript>23low</subscript></entry> | ||
| 59 | <entry>Y'<subscript>23high</subscript></entry> | ||
| 60 | </row> | ||
| 61 | <row> | ||
| 62 | <entry>start + 24:</entry> | ||
| 63 | <entry>Y'<subscript>30low</subscript></entry> | ||
| 64 | <entry>Y'<subscript>30high</subscript></entry> | ||
| 65 | <entry>Y'<subscript>31low</subscript></entry> | ||
| 66 | <entry>Y'<subscript>31high</subscript></entry> | ||
| 67 | <entry>Y'<subscript>32low</subscript></entry> | ||
| 68 | <entry>Y'<subscript>32high</subscript></entry> | ||
| 69 | <entry>Y'<subscript>33low</subscript></entry> | ||
| 70 | <entry>Y'<subscript>33high</subscript></entry> | ||
| 71 | </row> | ||
| 72 | </tbody> | ||
| 73 | </tgroup> | ||
| 74 | </informaltable> | ||
| 75 | </para> | ||
| 76 | </formalpara> | ||
| 77 | </example> | ||
| 78 | </refsect1> | ||
| 79 | </refentry> | ||
diff --git a/Documentation/DocBook/v4l/pixfmt.xml b/Documentation/DocBook/v4l/pixfmt.xml index c6fdcbbd1b41..40af4beb48b9 100644 --- a/Documentation/DocBook/v4l/pixfmt.xml +++ b/Documentation/DocBook/v4l/pixfmt.xml | |||
| @@ -696,6 +696,7 @@ information.</para> | |||
| 696 | &sub-packed-yuv; | 696 | &sub-packed-yuv; |
| 697 | &sub-grey; | 697 | &sub-grey; |
| 698 | &sub-y10; | 698 | &sub-y10; |
| 699 | &sub-y12; | ||
| 699 | &sub-y16; | 700 | &sub-y16; |
| 700 | &sub-yuyv; | 701 | &sub-yuyv; |
| 701 | &sub-uyvy; | 702 | &sub-uyvy; |
diff --git a/Documentation/DocBook/v4l/subdev-formats.xml b/Documentation/DocBook/v4l/subdev-formats.xml index 7041127d6dfc..d7ccd25edcc1 100644 --- a/Documentation/DocBook/v4l/subdev-formats.xml +++ b/Documentation/DocBook/v4l/subdev-formats.xml | |||
| @@ -456,6 +456,23 @@ | |||
| 456 | <entry>b<subscript>1</subscript></entry> | 456 | <entry>b<subscript>1</subscript></entry> |
| 457 | <entry>b<subscript>0</subscript></entry> | 457 | <entry>b<subscript>0</subscript></entry> |
| 458 | </row> | 458 | </row> |
| 459 | <row id="V4L2-MBUS-FMT-SGBRG8-1X8"> | ||
| 460 | <entry>V4L2_MBUS_FMT_SGBRG8_1X8</entry> | ||
| 461 | <entry>0x3013</entry> | ||
| 462 | <entry></entry> | ||
| 463 | <entry>-</entry> | ||
| 464 | <entry>-</entry> | ||
| 465 | <entry>-</entry> | ||
| 466 | <entry>-</entry> | ||
| 467 | <entry>g<subscript>7</subscript></entry> | ||
| 468 | <entry>g<subscript>6</subscript></entry> | ||
| 469 | <entry>g<subscript>5</subscript></entry> | ||
| 470 | <entry>g<subscript>4</subscript></entry> | ||
| 471 | <entry>g<subscript>3</subscript></entry> | ||
| 472 | <entry>g<subscript>2</subscript></entry> | ||
| 473 | <entry>g<subscript>1</subscript></entry> | ||
| 474 | <entry>g<subscript>0</subscript></entry> | ||
| 475 | </row> | ||
| 459 | <row id="V4L2-MBUS-FMT-SGRBG8-1X8"> | 476 | <row id="V4L2-MBUS-FMT-SGRBG8-1X8"> |
| 460 | <entry>V4L2_MBUS_FMT_SGRBG8_1X8</entry> | 477 | <entry>V4L2_MBUS_FMT_SGRBG8_1X8</entry> |
| 461 | <entry>0x3002</entry> | 478 | <entry>0x3002</entry> |
| @@ -473,6 +490,23 @@ | |||
| 473 | <entry>g<subscript>1</subscript></entry> | 490 | <entry>g<subscript>1</subscript></entry> |
| 474 | <entry>g<subscript>0</subscript></entry> | 491 | <entry>g<subscript>0</subscript></entry> |
| 475 | </row> | 492 | </row> |
| 493 | <row id="V4L2-MBUS-FMT-SRGGB8-1X8"> | ||
| 494 | <entry>V4L2_MBUS_FMT_SRGGB8_1X8</entry> | ||
| 495 | <entry>0x3014</entry> | ||
| 496 | <entry></entry> | ||
| 497 | <entry>-</entry> | ||
| 498 | <entry>-</entry> | ||
| 499 | <entry>-</entry> | ||
| 500 | <entry>-</entry> | ||
| 501 | <entry>r<subscript>7</subscript></entry> | ||
| 502 | <entry>r<subscript>6</subscript></entry> | ||
| 503 | <entry>r<subscript>5</subscript></entry> | ||
| 504 | <entry>r<subscript>4</subscript></entry> | ||
| 505 | <entry>r<subscript>3</subscript></entry> | ||
| 506 | <entry>r<subscript>2</subscript></entry> | ||
| 507 | <entry>r<subscript>1</subscript></entry> | ||
| 508 | <entry>r<subscript>0</subscript></entry> | ||
| 509 | </row> | ||
| 476 | <row id="V4L2-MBUS-FMT-SBGGR10-DPCM8-1X8"> | 510 | <row id="V4L2-MBUS-FMT-SBGGR10-DPCM8-1X8"> |
| 477 | <entry>V4L2_MBUS_FMT_SBGGR10_DPCM8_1X8</entry> | 511 | <entry>V4L2_MBUS_FMT_SBGGR10_DPCM8_1X8</entry> |
| 478 | <entry>0x300b</entry> | 512 | <entry>0x300b</entry> |
| @@ -2159,6 +2193,31 @@ | |||
| 2159 | <entry>u<subscript>1</subscript></entry> | 2193 | <entry>u<subscript>1</subscript></entry> |
| 2160 | <entry>u<subscript>0</subscript></entry> | 2194 | <entry>u<subscript>0</subscript></entry> |
| 2161 | </row> | 2195 | </row> |
| 2196 | <row id="V4L2-MBUS-FMT-Y12-1X12"> | ||
| 2197 | <entry>V4L2_MBUS_FMT_Y12_1X12</entry> | ||
| 2198 | <entry>0x2013</entry> | ||
| 2199 | <entry></entry> | ||
| 2200 | <entry>-</entry> | ||
| 2201 | <entry>-</entry> | ||
| 2202 | <entry>-</entry> | ||
| 2203 | <entry>-</entry> | ||
| 2204 | <entry>-</entry> | ||
| 2205 | <entry>-</entry> | ||
| 2206 | <entry>-</entry> | ||
| 2207 | <entry>-</entry> | ||
| 2208 | <entry>y<subscript>11</subscript></entry> | ||
| 2209 | <entry>y<subscript>10</subscript></entry> | ||
| 2210 | <entry>y<subscript>9</subscript></entry> | ||
| 2211 | <entry>y<subscript>8</subscript></entry> | ||
| 2212 | <entry>y<subscript>7</subscript></entry> | ||
| 2213 | <entry>y<subscript>6</subscript></entry> | ||
| 2214 | <entry>y<subscript>5</subscript></entry> | ||
| 2215 | <entry>y<subscript>4</subscript></entry> | ||
| 2216 | <entry>y<subscript>3</subscript></entry> | ||
| 2217 | <entry>y<subscript>2</subscript></entry> | ||
| 2218 | <entry>y<subscript>1</subscript></entry> | ||
| 2219 | <entry>y<subscript>0</subscript></entry> | ||
| 2220 | </row> | ||
| 2162 | <row id="V4L2-MBUS-FMT-UYVY8-1X16"> | 2221 | <row id="V4L2-MBUS-FMT-UYVY8-1X16"> |
| 2163 | <entry>V4L2_MBUS_FMT_UYVY8_1X16</entry> | 2222 | <entry>V4L2_MBUS_FMT_UYVY8_1X16</entry> |
| 2164 | <entry>0x200f</entry> | 2223 | <entry>0x200f</entry> |
diff --git a/Documentation/cgroups/memory.txt b/Documentation/cgroups/memory.txt index b6ed61c95856..7c163477fcd8 100644 --- a/Documentation/cgroups/memory.txt +++ b/Documentation/cgroups/memory.txt | |||
| @@ -52,8 +52,10 @@ Brief summary of control files. | |||
| 52 | tasks # attach a task(thread) and show list of threads | 52 | tasks # attach a task(thread) and show list of threads |
| 53 | cgroup.procs # show list of processes | 53 | cgroup.procs # show list of processes |
| 54 | cgroup.event_control # an interface for event_fd() | 54 | cgroup.event_control # an interface for event_fd() |
| 55 | memory.usage_in_bytes # show current memory(RSS+Cache) usage. | 55 | memory.usage_in_bytes # show current res_counter usage for memory |
| 56 | memory.memsw.usage_in_bytes # show current memory+Swap usage | 56 | (See 5.5 for details) |
| 57 | memory.memsw.usage_in_bytes # show current res_counter usage for memory+Swap | ||
| 58 | (See 5.5 for details) | ||
| 57 | memory.limit_in_bytes # set/show limit of memory usage | 59 | memory.limit_in_bytes # set/show limit of memory usage |
| 58 | memory.memsw.limit_in_bytes # set/show limit of memory+Swap usage | 60 | memory.memsw.limit_in_bytes # set/show limit of memory+Swap usage |
| 59 | memory.failcnt # show the number of memory usage hits limits | 61 | memory.failcnt # show the number of memory usage hits limits |
| @@ -453,6 +455,15 @@ memory under it will be reclaimed. | |||
| 453 | You can reset failcnt by writing 0 to failcnt file. | 455 | You can reset failcnt by writing 0 to failcnt file. |
| 454 | # echo 0 > .../memory.failcnt | 456 | # echo 0 > .../memory.failcnt |
| 455 | 457 | ||
| 458 | 5.5 usage_in_bytes | ||
| 459 | |||
| 460 | For efficiency, as other kernel components, memory cgroup uses some optimization | ||
| 461 | to avoid unnecessary cacheline false sharing. usage_in_bytes is affected by the | ||
| 462 | method and doesn't show 'exact' value of memory(and swap) usage, it's an fuzz | ||
| 463 | value for efficient access. (Of course, when necessary, it's synchronized.) | ||
| 464 | If you want to know more exact memory usage, you should use RSS+CACHE(+SWAP) | ||
| 465 | value in memory.stat(see 5.2). | ||
| 466 | |||
| 456 | 6. Hierarchy support | 467 | 6. Hierarchy support |
| 457 | 468 | ||
| 458 | The memory controller supports a deep hierarchy and hierarchical accounting. | 469 | The memory controller supports a deep hierarchy and hierarchical accounting. |
diff --git a/Documentation/hwmon/adm1021 b/Documentation/hwmon/adm1021 index 03d02bfb3df1..02ad96cf9b2b 100644 --- a/Documentation/hwmon/adm1021 +++ b/Documentation/hwmon/adm1021 | |||
| @@ -14,10 +14,6 @@ Supported chips: | |||
| 14 | Prefix: 'gl523sm' | 14 | Prefix: 'gl523sm' |
| 15 | Addresses scanned: I2C 0x18 - 0x1a, 0x29 - 0x2b, 0x4c - 0x4e | 15 | Addresses scanned: I2C 0x18 - 0x1a, 0x29 - 0x2b, 0x4c - 0x4e |
| 16 | Datasheet: | 16 | Datasheet: |
| 17 | * Intel Xeon Processor | ||
| 18 | Prefix: - any other - may require 'force_adm1021' parameter | ||
| 19 | Addresses scanned: none | ||
| 20 | Datasheet: Publicly available at Intel website | ||
| 21 | * Maxim MAX1617 | 17 | * Maxim MAX1617 |
| 22 | Prefix: 'max1617' | 18 | Prefix: 'max1617' |
| 23 | Addresses scanned: I2C 0x18 - 0x1a, 0x29 - 0x2b, 0x4c - 0x4e | 19 | Addresses scanned: I2C 0x18 - 0x1a, 0x29 - 0x2b, 0x4c - 0x4e |
| @@ -91,21 +87,27 @@ will do no harm, but will return 'old' values. It is possible to make | |||
| 91 | ADM1021-clones do faster measurements, but there is really no good reason | 87 | ADM1021-clones do faster measurements, but there is really no good reason |
| 92 | for that. | 88 | for that. |
| 93 | 89 | ||
| 94 | Xeon support | ||
| 95 | ------------ | ||
| 96 | 90 | ||
| 97 | Some Xeon processors have real max1617, adm1021, or compatible chips | 91 | Netburst-based Xeon support |
| 98 | within them, with two temperature sensors. | 92 | --------------------------- |
| 99 | 93 | ||
| 100 | Other Xeons have chips with only one sensor. | 94 | Some Xeon processors based on the Netburst (early Pentium 4, from 2001 to |
| 95 | 2003) microarchitecture had real MAX1617, ADM1021, or compatible chips | ||
| 96 | within them, with two temperature sensors. Other Xeon processors of this | ||
| 97 | era (with 400 MHz FSB) had chips with only one temperature sensor. | ||
| 101 | 98 | ||
| 102 | If you have a Xeon, and the adm1021 module loads, and both temperatures | 99 | If you have such an old Xeon, and you get two valid temperatures when |
| 103 | appear valid, then things are good. | 100 | loading the adm1021 module, then things are good. |
| 104 | 101 | ||
| 105 | If the adm1021 module doesn't load, you should try this: | 102 | If nothing happens when loading the adm1021 module, and you are certain |
| 106 | modprobe adm1021 force_adm1021=BUS,ADDRESS | 103 | that your specific Xeon processor model includes compatible sensors, you |
| 107 | ADDRESS can only be 0x18, 0x1a, 0x29, 0x2b, 0x4c, or 0x4e. | 104 | will have to explicitly instantiate the sensor chips from user-space. See |
| 105 | method 4 in Documentation/i2c/instantiating-devices. Possible slave | ||
| 106 | addresses are 0x18, 0x1a, 0x29, 0x2b, 0x4c, or 0x4e. It is likely that | ||
| 107 | only temp2 will be correct and temp1 will have to be ignored. | ||
| 108 | 108 | ||
| 109 | If you have dual Xeons you may have appear to have two separate | 109 | Previous generations of the Xeon processor (based on Pentium II/III) |
| 110 | adm1021-compatible chips, or two single-temperature sensors, at distinct | 110 | didn't have these sensors. Next generations of Xeon processors (533 MHz |
| 111 | addresses. | 111 | FSB and faster) lost them, until the Core-based generation which |
| 112 | introduced integrated digital thermal sensors. These are supported by | ||
| 113 | the coretemp driver. | ||
diff --git a/Documentation/hwmon/lm90 b/Documentation/hwmon/lm90 index fa475c0a48a3..f3efd18e87f4 100644 --- a/Documentation/hwmon/lm90 +++ b/Documentation/hwmon/lm90 | |||
| @@ -32,6 +32,16 @@ Supported chips: | |||
| 32 | Addresses scanned: I2C 0x4c and 0x4d | 32 | Addresses scanned: I2C 0x4c and 0x4d |
| 33 | Datasheet: Publicly available at the ON Semiconductor website | 33 | Datasheet: Publicly available at the ON Semiconductor website |
| 34 | http://www.onsemi.com/PowerSolutions/product.do?id=ADT7461 | 34 | http://www.onsemi.com/PowerSolutions/product.do?id=ADT7461 |
| 35 | * Analog Devices ADT7461A | ||
| 36 | Prefix: 'adt7461a' | ||
| 37 | Addresses scanned: I2C 0x4c and 0x4d | ||
| 38 | Datasheet: Publicly available at the ON Semiconductor website | ||
| 39 | http://www.onsemi.com/PowerSolutions/product.do?id=ADT7461A | ||
| 40 | * ON Semiconductor NCT1008 | ||
| 41 | Prefix: 'nct1008' | ||
| 42 | Addresses scanned: I2C 0x4c and 0x4d | ||
| 43 | Datasheet: Publicly available at the ON Semiconductor website | ||
| 44 | http://www.onsemi.com/PowerSolutions/product.do?id=NCT1008 | ||
| 35 | * Maxim MAX6646 | 45 | * Maxim MAX6646 |
| 36 | Prefix: 'max6646' | 46 | Prefix: 'max6646' |
| 37 | Addresses scanned: I2C 0x4d | 47 | Addresses scanned: I2C 0x4d |
| @@ -149,7 +159,7 @@ ADM1032: | |||
| 149 | * ALERT is triggered by open remote sensor. | 159 | * ALERT is triggered by open remote sensor. |
| 150 | * SMBus PEC support for Write Byte and Receive Byte transactions. | 160 | * SMBus PEC support for Write Byte and Receive Byte transactions. |
| 151 | 161 | ||
| 152 | ADT7461: | 162 | ADT7461, ADT7461A, NCT1008: |
| 153 | * Extended temperature range (breaks compatibility) | 163 | * Extended temperature range (breaks compatibility) |
| 154 | * Lower resolution for remote temperature | 164 | * Lower resolution for remote temperature |
| 155 | 165 | ||
| @@ -195,9 +205,9 @@ are exported, one for each channel, but these values are of course linked. | |||
| 195 | Only the local hysteresis can be set from user-space, and the same delta | 205 | Only the local hysteresis can be set from user-space, and the same delta |
| 196 | applies to the remote hysteresis. | 206 | applies to the remote hysteresis. |
| 197 | 207 | ||
| 198 | The lm90 driver will not update its values more frequently than every | 208 | The lm90 driver will not update its values more frequently than configured with |
| 199 | other second; reading them more often will do no harm, but will return | 209 | the update_interval attribute; reading them more often will do no harm, but will |
| 200 | 'old' values. | 210 | return 'old' values. |
| 201 | 211 | ||
| 202 | SMBus Alert Support | 212 | SMBus Alert Support |
| 203 | ------------------- | 213 | ------------------- |
| @@ -205,11 +215,12 @@ SMBus Alert Support | |||
| 205 | This driver has basic support for SMBus alert. When an alert is received, | 215 | This driver has basic support for SMBus alert. When an alert is received, |
| 206 | the status register is read and the faulty temperature channel is logged. | 216 | the status register is read and the faulty temperature channel is logged. |
| 207 | 217 | ||
| 208 | The Analog Devices chips (ADM1032 and ADT7461) do not implement the SMBus | 218 | The Analog Devices chips (ADM1032, ADT7461 and ADT7461A) and ON |
| 209 | alert protocol properly so additional care is needed: the ALERT output is | 219 | Semiconductor chips (NCT1008) do not implement the SMBus alert protocol |
| 210 | disabled when an alert is received, and is re-enabled only when the alarm | 220 | properly so additional care is needed: the ALERT output is disabled when |
| 211 | is gone. Otherwise the chip would block alerts from other chips in the bus | 221 | an alert is received, and is re-enabled only when the alarm is gone. |
| 212 | as long as the alarm is active. | 222 | Otherwise the chip would block alerts from other chips in the bus as long |
| 223 | as the alarm is active. | ||
| 213 | 224 | ||
| 214 | PEC Support | 225 | PEC Support |
| 215 | ----------- | 226 | ----------- |
diff --git a/Documentation/video4linux/sh_mobile_ceu_camera.txt b/Documentation/video4linux/sh_mobile_ceu_camera.txt index cb47e723af74..1e96ce6e2d2f 100644 --- a/Documentation/video4linux/sh_mobile_ceu_camera.txt +++ b/Documentation/video4linux/sh_mobile_ceu_camera.txt | |||
| @@ -37,7 +37,7 @@ Generic scaling / cropping scheme | |||
| 37 | -1'- | 37 | -1'- |
| 38 | 38 | ||
| 39 | In the above chart minuses and slashes represent "real" data amounts, points and | 39 | In the above chart minuses and slashes represent "real" data amounts, points and |
| 40 | accents represent "useful" data, basically, CEU scaled amd cropped output, | 40 | accents represent "useful" data, basically, CEU scaled and cropped output, |
| 41 | mapped back onto the client's source plane. | 41 | mapped back onto the client's source plane. |
| 42 | 42 | ||
| 43 | Such a configuration can be produced by user requests: | 43 | Such a configuration can be produced by user requests: |
| @@ -65,7 +65,7 @@ Do not touch input rectangle - it is already optimal. | |||
| 65 | 65 | ||
| 66 | 1. Calculate current sensor scales: | 66 | 1. Calculate current sensor scales: |
| 67 | 67 | ||
| 68 | scale_s = ((3') - (3)) / ((2') - (2)) | 68 | scale_s = ((2') - (2)) / ((3') - (3)) |
| 69 | 69 | ||
| 70 | 2. Calculate "effective" input crop (sensor subwindow) - CEU crop scaled back at | 70 | 2. Calculate "effective" input crop (sensor subwindow) - CEU crop scaled back at |
| 71 | current sensor scales onto input window - this is user S_CROP: | 71 | current sensor scales onto input window - this is user S_CROP: |
| @@ -80,7 +80,7 @@ window: | |||
| 80 | 4. Calculate sensor output window by applying combined scales to real input | 80 | 4. Calculate sensor output window by applying combined scales to real input |
| 81 | window: | 81 | window: |
| 82 | 82 | ||
| 83 | width_s_out = ((2') - (2)) / scale_comb | 83 | width_s_out = ((7') - (7)) = ((2') - (2)) / scale_comb |
| 84 | 84 | ||
| 85 | 5. Apply iterative sensor S_FMT for sensor output window. | 85 | 5. Apply iterative sensor S_FMT for sensor output window. |
| 86 | 86 | ||
diff --git a/Documentation/workqueue.txt b/Documentation/workqueue.txt index 01c513fac40e..a0b577de918f 100644 --- a/Documentation/workqueue.txt +++ b/Documentation/workqueue.txt | |||
| @@ -12,6 +12,7 @@ CONTENTS | |||
| 12 | 4. Application Programming Interface (API) | 12 | 4. Application Programming Interface (API) |
| 13 | 5. Example Execution Scenarios | 13 | 5. Example Execution Scenarios |
| 14 | 6. Guidelines | 14 | 6. Guidelines |
| 15 | 7. Debugging | ||
| 15 | 16 | ||
| 16 | 17 | ||
| 17 | 1. Introduction | 18 | 1. Introduction |
| @@ -379,3 +380,42 @@ If q1 has WQ_CPU_INTENSIVE set, | |||
| 379 | * Unless work items are expected to consume a huge amount of CPU | 380 | * Unless work items are expected to consume a huge amount of CPU |
| 380 | cycles, using a bound wq is usually beneficial due to the increased | 381 | cycles, using a bound wq is usually beneficial due to the increased |
| 381 | level of locality in wq operations and work item execution. | 382 | level of locality in wq operations and work item execution. |
| 383 | |||
| 384 | |||
| 385 | 7. Debugging | ||
| 386 | |||
| 387 | Because the work functions are executed by generic worker threads | ||
| 388 | there are a few tricks needed to shed some light on misbehaving | ||
| 389 | workqueue users. | ||
| 390 | |||
| 391 | Worker threads show up in the process list as: | ||
| 392 | |||
| 393 | root 5671 0.0 0.0 0 0 ? S 12:07 0:00 [kworker/0:1] | ||
| 394 | root 5672 0.0 0.0 0 0 ? S 12:07 0:00 [kworker/1:2] | ||
| 395 | root 5673 0.0 0.0 0 0 ? S 12:12 0:00 [kworker/0:0] | ||
| 396 | root 5674 0.0 0.0 0 0 ? S 12:13 0:00 [kworker/1:0] | ||
| 397 | |||
| 398 | If kworkers are going crazy (using too much cpu), there are two types | ||
| 399 | of possible problems: | ||
| 400 | |||
| 401 | 1. Something beeing scheduled in rapid succession | ||
| 402 | 2. A single work item that consumes lots of cpu cycles | ||
| 403 | |||
| 404 | The first one can be tracked using tracing: | ||
| 405 | |||
| 406 | $ echo workqueue:workqueue_queue_work > /sys/kernel/debug/tracing/set_event | ||
| 407 | $ cat /sys/kernel/debug/tracing/trace_pipe > out.txt | ||
| 408 | (wait a few secs) | ||
| 409 | ^C | ||
| 410 | |||
| 411 | If something is busy looping on work queueing, it would be dominating | ||
| 412 | the output and the offender can be determined with the work item | ||
| 413 | function. | ||
| 414 | |||
| 415 | For the second type of problems it should be possible to just check | ||
| 416 | the stack trace of the offending worker thread. | ||
| 417 | |||
| 418 | $ cat /proc/THE_OFFENDING_KWORKER/stack | ||
| 419 | |||
| 420 | The work item's function should be trivially visible in the stack | ||
| 421 | trace. | ||
diff --git a/MAINTAINERS b/MAINTAINERS index 13803127b68f..ae65b29ad936 100644 --- a/MAINTAINERS +++ b/MAINTAINERS | |||
| @@ -1032,12 +1032,13 @@ W: http://www.fluff.org/ben/linux/ | |||
| 1032 | S: Maintained | 1032 | S: Maintained |
| 1033 | F: arch/arm/mach-s3c64xx/ | 1033 | F: arch/arm/mach-s3c64xx/ |
| 1034 | 1034 | ||
| 1035 | ARM/S5P ARM ARCHITECTURES | 1035 | ARM/S5P EXYNOS ARM ARCHITECTURES |
| 1036 | M: Kukjin Kim <kgene.kim@samsung.com> | 1036 | M: Kukjin Kim <kgene.kim@samsung.com> |
| 1037 | L: linux-arm-kernel@lists.infradead.org (moderated for non-subscribers) | 1037 | L: linux-arm-kernel@lists.infradead.org (moderated for non-subscribers) |
| 1038 | L: linux-samsung-soc@vger.kernel.org (moderated for non-subscribers) | 1038 | L: linux-samsung-soc@vger.kernel.org (moderated for non-subscribers) |
| 1039 | S: Maintained | 1039 | S: Maintained |
| 1040 | F: arch/arm/mach-s5p*/ | 1040 | F: arch/arm/mach-s5p*/ |
| 1041 | F: arch/arm/mach-exynos*/ | ||
| 1041 | 1042 | ||
| 1042 | ARM/SAMSUNG MOBILE MACHINE SUPPORT | 1043 | ARM/SAMSUNG MOBILE MACHINE SUPPORT |
| 1043 | M: Kyungmin Park <kyungmin.park@samsung.com> | 1044 | M: Kyungmin Park <kyungmin.park@samsung.com> |
| @@ -2250,10 +2251,10 @@ F: drivers/gpu/drm/ | |||
| 2250 | F: include/drm/ | 2251 | F: include/drm/ |
| 2251 | 2252 | ||
| 2252 | INTEL DRM DRIVERS (excluding Poulsbo, Moorestown and derivative chipsets) | 2253 | INTEL DRM DRIVERS (excluding Poulsbo, Moorestown and derivative chipsets) |
| 2253 | M: Chris Wilson <chris@chris-wilson.co.uk> | 2254 | M: Keith Packard <keithp@keithp.com> |
| 2254 | L: intel-gfx@lists.freedesktop.org (subscribers-only) | 2255 | L: intel-gfx@lists.freedesktop.org (subscribers-only) |
| 2255 | L: dri-devel@lists.freedesktop.org | 2256 | L: dri-devel@lists.freedesktop.org |
| 2256 | T: git git://git.kernel.org/pub/scm/linux/kernel/git/ickle/drm-intel.git | 2257 | T: git git://git.kernel.org/pub/scm/linux/kernel/git/keithp/linux-2.6.git |
| 2257 | S: Supported | 2258 | S: Supported |
| 2258 | F: drivers/gpu/drm/i915 | 2259 | F: drivers/gpu/drm/i915 |
| 2259 | F: include/drm/i915* | 2260 | F: include/drm/i915* |
| @@ -2808,7 +2809,7 @@ GPIO SUBSYSTEM | |||
| 2808 | M: Grant Likely <grant.likely@secretlab.ca> | 2809 | M: Grant Likely <grant.likely@secretlab.ca> |
| 2809 | S: Maintained | 2810 | S: Maintained |
| 2810 | T: git git://git.secretlab.ca/git/linux-2.6.git | 2811 | T: git git://git.secretlab.ca/git/linux-2.6.git |
| 2811 | F: Documentation/gpio/gpio.txt | 2812 | F: Documentation/gpio.txt |
| 2812 | F: drivers/gpio/ | 2813 | F: drivers/gpio/ |
| 2813 | F: include/linux/gpio* | 2814 | F: include/linux/gpio* |
| 2814 | 2815 | ||
| @@ -6555,7 +6556,7 @@ S: Maintained | |||
| 6555 | F: drivers/usb/host/uhci* | 6556 | F: drivers/usb/host/uhci* |
| 6556 | 6557 | ||
| 6557 | USB "USBNET" DRIVER FRAMEWORK | 6558 | USB "USBNET" DRIVER FRAMEWORK |
| 6558 | M: David Brownell <dbrownell@users.sourceforge.net> | 6559 | M: Oliver Neukum <oneukum@suse.de> |
| 6559 | L: netdev@vger.kernel.org | 6560 | L: netdev@vger.kernel.org |
| 6560 | W: http://www.linux-usb.org/usbnet | 6561 | W: http://www.linux-usb.org/usbnet |
| 6561 | S: Maintained | 6562 | S: Maintained |
| @@ -6921,6 +6922,18 @@ T: git git://git.kernel.org/pub/scm/linux/kernel/git/mjg59/platform-drivers-x86. | |||
| 6921 | S: Maintained | 6922 | S: Maintained |
| 6922 | F: drivers/platform/x86 | 6923 | F: drivers/platform/x86 |
| 6923 | 6924 | ||
| 6925 | XEN HYPERVISOR INTERFACE | ||
| 6926 | M: Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com> | ||
| 6927 | M: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com> | ||
| 6928 | L: xen-devel@lists.xensource.com (moderated for non-subscribers) | ||
| 6929 | L: virtualization@lists.linux-foundation.org | ||
| 6930 | S: Supported | ||
| 6931 | F: arch/x86/xen/ | ||
| 6932 | F: drivers/*/xen-*front.c | ||
| 6933 | F: drivers/xen/ | ||
| 6934 | F: arch/x86/include/asm/xen/ | ||
| 6935 | F: include/xen/ | ||
| 6936 | |||
| 6924 | XEN NETWORK BACKEND DRIVER | 6937 | XEN NETWORK BACKEND DRIVER |
| 6925 | M: Ian Campbell <ian.campbell@citrix.com> | 6938 | M: Ian Campbell <ian.campbell@citrix.com> |
| 6926 | L: xen-devel@lists.xensource.com (moderated for non-subscribers) | 6939 | L: xen-devel@lists.xensource.com (moderated for non-subscribers) |
| @@ -6942,18 +6955,6 @@ S: Supported | |||
| 6942 | F: arch/x86/xen/*swiotlb* | 6955 | F: arch/x86/xen/*swiotlb* |
| 6943 | F: drivers/xen/*swiotlb* | 6956 | F: drivers/xen/*swiotlb* |
| 6944 | 6957 | ||
| 6945 | XEN HYPERVISOR INTERFACE | ||
| 6946 | M: Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com> | ||
| 6947 | M: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com> | ||
| 6948 | L: xen-devel@lists.xensource.com (moderated for non-subscribers) | ||
| 6949 | L: virtualization@lists.linux-foundation.org | ||
| 6950 | S: Supported | ||
| 6951 | F: arch/x86/xen/ | ||
| 6952 | F: drivers/*/xen-*front.c | ||
| 6953 | F: drivers/xen/ | ||
| 6954 | F: arch/x86/include/asm/xen/ | ||
| 6955 | F: include/xen/ | ||
| 6956 | |||
| 6957 | XFS FILESYSTEM | 6958 | XFS FILESYSTEM |
| 6958 | P: Silicon Graphics Inc | 6959 | P: Silicon Graphics Inc |
| 6959 | M: Alex Elder <aelder@sgi.com> | 6960 | M: Alex Elder <aelder@sgi.com> |
| @@ -1,7 +1,7 @@ | |||
| 1 | VERSION = 2 | 1 | VERSION = 2 |
| 2 | PATCHLEVEL = 6 | 2 | PATCHLEVEL = 6 |
| 3 | SUBLEVEL = 39 | 3 | SUBLEVEL = 39 |
| 4 | EXTRAVERSION = -rc5 | 4 | EXTRAVERSION = -rc6 |
| 5 | NAME = Flesh-Eating Bats with Fangs | 5 | NAME = Flesh-Eating Bats with Fangs |
| 6 | 6 | ||
| 7 | # *DOCUMENTATION* | 7 | # *DOCUMENTATION* |
diff --git a/arch/arm/include/asm/kprobes.h b/arch/arm/include/asm/kprobes.h index bb8a19bd5822..e46bdd0097eb 100644 --- a/arch/arm/include/asm/kprobes.h +++ b/arch/arm/include/asm/kprobes.h | |||
| @@ -39,10 +39,13 @@ typedef u32 kprobe_opcode_t; | |||
| 39 | struct kprobe; | 39 | struct kprobe; |
| 40 | typedef void (kprobe_insn_handler_t)(struct kprobe *, struct pt_regs *); | 40 | typedef void (kprobe_insn_handler_t)(struct kprobe *, struct pt_regs *); |
| 41 | 41 | ||
| 42 | typedef unsigned long (kprobe_check_cc)(unsigned long); | ||
| 43 | |||
| 42 | /* Architecture specific copy of original instruction. */ | 44 | /* Architecture specific copy of original instruction. */ |
| 43 | struct arch_specific_insn { | 45 | struct arch_specific_insn { |
| 44 | kprobe_opcode_t *insn; | 46 | kprobe_opcode_t *insn; |
| 45 | kprobe_insn_handler_t *insn_handler; | 47 | kprobe_insn_handler_t *insn_handler; |
| 48 | kprobe_check_cc *insn_check_cc; | ||
| 46 | }; | 49 | }; |
| 47 | 50 | ||
| 48 | struct prev_kprobe { | 51 | struct prev_kprobe { |
diff --git a/arch/arm/kernel/kprobes-decode.c b/arch/arm/kernel/kprobes-decode.c index 23891317dc4b..15eeff6aea0e 100644 --- a/arch/arm/kernel/kprobes-decode.c +++ b/arch/arm/kernel/kprobes-decode.c | |||
| @@ -34,9 +34,6 @@ | |||
| 34 | * | 34 | * |
| 35 | * *) If the PC is written to by the instruction, the | 35 | * *) If the PC is written to by the instruction, the |
| 36 | * instruction must be fully simulated in software. | 36 | * instruction must be fully simulated in software. |
| 37 | * If it is a conditional instruction, the handler | ||
| 38 | * will use insn[0] to copy its condition code to | ||
| 39 | * set r0 to 1 and insn[1] to "mov pc, lr" to return. | ||
| 40 | * | 37 | * |
| 41 | * *) Otherwise, a modified form of the instruction is | 38 | * *) Otherwise, a modified form of the instruction is |
| 42 | * directly executed. Its handler calls the | 39 | * directly executed. Its handler calls the |
| @@ -68,13 +65,17 @@ | |||
| 68 | 65 | ||
| 69 | #define branch_displacement(insn) sign_extend(((insn) & 0xffffff) << 2, 25) | 66 | #define branch_displacement(insn) sign_extend(((insn) & 0xffffff) << 2, 25) |
| 70 | 67 | ||
| 68 | #define is_r15(insn, bitpos) (((insn) & (0xf << bitpos)) == (0xf << bitpos)) | ||
| 69 | |||
| 70 | /* | ||
| 71 | * Test if load/store instructions writeback the address register. | ||
| 72 | * if P (bit 24) == 0 or W (bit 21) == 1 | ||
| 73 | */ | ||
| 74 | #define is_writeback(insn) ((insn ^ 0x01000000) & 0x01200000) | ||
| 75 | |||
| 71 | #define PSR_fs (PSR_f|PSR_s) | 76 | #define PSR_fs (PSR_f|PSR_s) |
| 72 | 77 | ||
| 73 | #define KPROBE_RETURN_INSTRUCTION 0xe1a0f00e /* mov pc, lr */ | 78 | #define KPROBE_RETURN_INSTRUCTION 0xe1a0f00e /* mov pc, lr */ |
| 74 | #define SET_R0_TRUE_INSTRUCTION 0xe3a00001 /* mov r0, #1 */ | ||
| 75 | |||
| 76 | #define truecc_insn(insn) (((insn) & 0xf0000000) | \ | ||
| 77 | (SET_R0_TRUE_INSTRUCTION & 0x0fffffff)) | ||
| 78 | 79 | ||
| 79 | typedef long (insn_0arg_fn_t)(void); | 80 | typedef long (insn_0arg_fn_t)(void); |
| 80 | typedef long (insn_1arg_fn_t)(long); | 81 | typedef long (insn_1arg_fn_t)(long); |
| @@ -419,14 +420,10 @@ insnslot_llret_4arg_rwflags(long r0, long r1, long r2, long r3, long *cpsr, | |||
| 419 | 420 | ||
| 420 | static void __kprobes simulate_bbl(struct kprobe *p, struct pt_regs *regs) | 421 | static void __kprobes simulate_bbl(struct kprobe *p, struct pt_regs *regs) |
| 421 | { | 422 | { |
| 422 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | ||
| 423 | kprobe_opcode_t insn = p->opcode; | 423 | kprobe_opcode_t insn = p->opcode; |
| 424 | long iaddr = (long)p->addr; | 424 | long iaddr = (long)p->addr; |
| 425 | int disp = branch_displacement(insn); | 425 | int disp = branch_displacement(insn); |
| 426 | 426 | ||
| 427 | if (!insnslot_1arg_rflags(0, regs->ARM_cpsr, i_fn)) | ||
| 428 | return; | ||
| 429 | |||
| 430 | if (insn & (1 << 24)) | 427 | if (insn & (1 << 24)) |
| 431 | regs->ARM_lr = iaddr + 4; | 428 | regs->ARM_lr = iaddr + 4; |
| 432 | 429 | ||
| @@ -446,14 +443,10 @@ static void __kprobes simulate_blx1(struct kprobe *p, struct pt_regs *regs) | |||
| 446 | 443 | ||
| 447 | static void __kprobes simulate_blx2bx(struct kprobe *p, struct pt_regs *regs) | 444 | static void __kprobes simulate_blx2bx(struct kprobe *p, struct pt_regs *regs) |
| 448 | { | 445 | { |
| 449 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | ||
| 450 | kprobe_opcode_t insn = p->opcode; | 446 | kprobe_opcode_t insn = p->opcode; |
| 451 | int rm = insn & 0xf; | 447 | int rm = insn & 0xf; |
| 452 | long rmv = regs->uregs[rm]; | 448 | long rmv = regs->uregs[rm]; |
| 453 | 449 | ||
| 454 | if (!insnslot_1arg_rflags(0, regs->ARM_cpsr, i_fn)) | ||
| 455 | return; | ||
| 456 | |||
| 457 | if (insn & (1 << 5)) | 450 | if (insn & (1 << 5)) |
| 458 | regs->ARM_lr = (long)p->addr + 4; | 451 | regs->ARM_lr = (long)p->addr + 4; |
| 459 | 452 | ||
| @@ -463,9 +456,16 @@ static void __kprobes simulate_blx2bx(struct kprobe *p, struct pt_regs *regs) | |||
| 463 | regs->ARM_cpsr |= PSR_T_BIT; | 456 | regs->ARM_cpsr |= PSR_T_BIT; |
| 464 | } | 457 | } |
| 465 | 458 | ||
| 459 | static void __kprobes simulate_mrs(struct kprobe *p, struct pt_regs *regs) | ||
| 460 | { | ||
| 461 | kprobe_opcode_t insn = p->opcode; | ||
| 462 | int rd = (insn >> 12) & 0xf; | ||
| 463 | unsigned long mask = 0xf8ff03df; /* Mask out execution state */ | ||
| 464 | regs->uregs[rd] = regs->ARM_cpsr & mask; | ||
| 465 | } | ||
| 466 | |||
| 466 | static void __kprobes simulate_ldm1stm1(struct kprobe *p, struct pt_regs *regs) | 467 | static void __kprobes simulate_ldm1stm1(struct kprobe *p, struct pt_regs *regs) |
| 467 | { | 468 | { |
| 468 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | ||
| 469 | kprobe_opcode_t insn = p->opcode; | 469 | kprobe_opcode_t insn = p->opcode; |
| 470 | int rn = (insn >> 16) & 0xf; | 470 | int rn = (insn >> 16) & 0xf; |
| 471 | int lbit = insn & (1 << 20); | 471 | int lbit = insn & (1 << 20); |
| @@ -476,9 +476,6 @@ static void __kprobes simulate_ldm1stm1(struct kprobe *p, struct pt_regs *regs) | |||
| 476 | int reg_bit_vector; | 476 | int reg_bit_vector; |
| 477 | int reg_count; | 477 | int reg_count; |
| 478 | 478 | ||
| 479 | if (!insnslot_1arg_rflags(0, regs->ARM_cpsr, i_fn)) | ||
| 480 | return; | ||
| 481 | |||
| 482 | reg_count = 0; | 479 | reg_count = 0; |
| 483 | reg_bit_vector = insn & 0xffff; | 480 | reg_bit_vector = insn & 0xffff; |
| 484 | while (reg_bit_vector) { | 481 | while (reg_bit_vector) { |
| @@ -510,11 +507,6 @@ static void __kprobes simulate_ldm1stm1(struct kprobe *p, struct pt_regs *regs) | |||
| 510 | 507 | ||
| 511 | static void __kprobes simulate_stm1_pc(struct kprobe *p, struct pt_regs *regs) | 508 | static void __kprobes simulate_stm1_pc(struct kprobe *p, struct pt_regs *regs) |
| 512 | { | 509 | { |
| 513 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | ||
| 514 | |||
| 515 | if (!insnslot_1arg_rflags(0, regs->ARM_cpsr, i_fn)) | ||
| 516 | return; | ||
| 517 | |||
| 518 | regs->ARM_pc = (long)p->addr + str_pc_offset; | 510 | regs->ARM_pc = (long)p->addr + str_pc_offset; |
| 519 | simulate_ldm1stm1(p, regs); | 511 | simulate_ldm1stm1(p, regs); |
| 520 | regs->ARM_pc = (long)p->addr + 4; | 512 | regs->ARM_pc = (long)p->addr + 4; |
| @@ -525,24 +517,16 @@ static void __kprobes simulate_mov_ipsp(struct kprobe *p, struct pt_regs *regs) | |||
| 525 | regs->uregs[12] = regs->uregs[13]; | 517 | regs->uregs[12] = regs->uregs[13]; |
| 526 | } | 518 | } |
| 527 | 519 | ||
| 528 | static void __kprobes emulate_ldcstc(struct kprobe *p, struct pt_regs *regs) | ||
| 529 | { | ||
| 530 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | ||
| 531 | kprobe_opcode_t insn = p->opcode; | ||
| 532 | int rn = (insn >> 16) & 0xf; | ||
| 533 | long rnv = regs->uregs[rn]; | ||
| 534 | |||
| 535 | /* Save Rn in case of writeback. */ | ||
| 536 | regs->uregs[rn] = insnslot_1arg_rflags(rnv, regs->ARM_cpsr, i_fn); | ||
| 537 | } | ||
| 538 | |||
| 539 | static void __kprobes emulate_ldrd(struct kprobe *p, struct pt_regs *regs) | 520 | static void __kprobes emulate_ldrd(struct kprobe *p, struct pt_regs *regs) |
| 540 | { | 521 | { |
| 541 | insn_2arg_fn_t *i_fn = (insn_2arg_fn_t *)&p->ainsn.insn[0]; | 522 | insn_2arg_fn_t *i_fn = (insn_2arg_fn_t *)&p->ainsn.insn[0]; |
| 542 | kprobe_opcode_t insn = p->opcode; | 523 | kprobe_opcode_t insn = p->opcode; |
| 524 | long ppc = (long)p->addr + 8; | ||
| 543 | int rd = (insn >> 12) & 0xf; | 525 | int rd = (insn >> 12) & 0xf; |
| 544 | int rn = (insn >> 16) & 0xf; | 526 | int rn = (insn >> 16) & 0xf; |
| 545 | int rm = insn & 0xf; /* rm may be invalid, don't care. */ | 527 | int rm = insn & 0xf; /* rm may be invalid, don't care. */ |
| 528 | long rmv = (rm == 15) ? ppc : regs->uregs[rm]; | ||
| 529 | long rnv = (rn == 15) ? ppc : regs->uregs[rn]; | ||
| 546 | 530 | ||
| 547 | /* Not following the C calling convention here, so need asm(). */ | 531 | /* Not following the C calling convention here, so need asm(). */ |
| 548 | __asm__ __volatile__ ( | 532 | __asm__ __volatile__ ( |
| @@ -554,29 +538,36 @@ static void __kprobes emulate_ldrd(struct kprobe *p, struct pt_regs *regs) | |||
| 554 | "str r0, %[rn] \n\t" /* in case of writeback */ | 538 | "str r0, %[rn] \n\t" /* in case of writeback */ |
| 555 | "str r2, %[rd0] \n\t" | 539 | "str r2, %[rd0] \n\t" |
| 556 | "str r3, %[rd1] \n\t" | 540 | "str r3, %[rd1] \n\t" |
| 557 | : [rn] "+m" (regs->uregs[rn]), | 541 | : [rn] "+m" (rnv), |
| 558 | [rd0] "=m" (regs->uregs[rd]), | 542 | [rd0] "=m" (regs->uregs[rd]), |
| 559 | [rd1] "=m" (regs->uregs[rd+1]) | 543 | [rd1] "=m" (regs->uregs[rd+1]) |
| 560 | : [rm] "m" (regs->uregs[rm]), | 544 | : [rm] "m" (rmv), |
| 561 | [cpsr] "r" (regs->ARM_cpsr), | 545 | [cpsr] "r" (regs->ARM_cpsr), |
| 562 | [i_fn] "r" (i_fn) | 546 | [i_fn] "r" (i_fn) |
| 563 | : "r0", "r1", "r2", "r3", "lr", "cc" | 547 | : "r0", "r1", "r2", "r3", "lr", "cc" |
| 564 | ); | 548 | ); |
| 549 | if (is_writeback(insn)) | ||
| 550 | regs->uregs[rn] = rnv; | ||
| 565 | } | 551 | } |
| 566 | 552 | ||
| 567 | static void __kprobes emulate_strd(struct kprobe *p, struct pt_regs *regs) | 553 | static void __kprobes emulate_strd(struct kprobe *p, struct pt_regs *regs) |
| 568 | { | 554 | { |
| 569 | insn_4arg_fn_t *i_fn = (insn_4arg_fn_t *)&p->ainsn.insn[0]; | 555 | insn_4arg_fn_t *i_fn = (insn_4arg_fn_t *)&p->ainsn.insn[0]; |
| 570 | kprobe_opcode_t insn = p->opcode; | 556 | kprobe_opcode_t insn = p->opcode; |
| 557 | long ppc = (long)p->addr + 8; | ||
| 571 | int rd = (insn >> 12) & 0xf; | 558 | int rd = (insn >> 12) & 0xf; |
| 572 | int rn = (insn >> 16) & 0xf; | 559 | int rn = (insn >> 16) & 0xf; |
| 573 | int rm = insn & 0xf; | 560 | int rm = insn & 0xf; |
| 574 | long rnv = regs->uregs[rn]; | 561 | long rnv = (rn == 15) ? ppc : regs->uregs[rn]; |
| 575 | long rmv = regs->uregs[rm]; /* rm/rmv may be invalid, don't care. */ | 562 | /* rm/rmv may be invalid, don't care. */ |
| 563 | long rmv = (rm == 15) ? ppc : regs->uregs[rm]; | ||
| 564 | long rnv_wb; | ||
| 576 | 565 | ||
| 577 | regs->uregs[rn] = insnslot_4arg_rflags(rnv, rmv, regs->uregs[rd], | 566 | rnv_wb = insnslot_4arg_rflags(rnv, rmv, regs->uregs[rd], |
| 578 | regs->uregs[rd+1], | 567 | regs->uregs[rd+1], |
| 579 | regs->ARM_cpsr, i_fn); | 568 | regs->ARM_cpsr, i_fn); |
| 569 | if (is_writeback(insn)) | ||
| 570 | regs->uregs[rn] = rnv_wb; | ||
| 580 | } | 571 | } |
| 581 | 572 | ||
| 582 | static void __kprobes emulate_ldr(struct kprobe *p, struct pt_regs *regs) | 573 | static void __kprobes emulate_ldr(struct kprobe *p, struct pt_regs *regs) |
| @@ -630,31 +621,6 @@ static void __kprobes emulate_str(struct kprobe *p, struct pt_regs *regs) | |||
| 630 | regs->uregs[rn] = rnv_wb; /* Save Rn in case of writeback. */ | 621 | regs->uregs[rn] = rnv_wb; /* Save Rn in case of writeback. */ |
| 631 | } | 622 | } |
| 632 | 623 | ||
| 633 | static void __kprobes emulate_mrrc(struct kprobe *p, struct pt_regs *regs) | ||
| 634 | { | ||
| 635 | insn_llret_0arg_fn_t *i_fn = (insn_llret_0arg_fn_t *)&p->ainsn.insn[0]; | ||
| 636 | kprobe_opcode_t insn = p->opcode; | ||
| 637 | union reg_pair fnr; | ||
| 638 | int rd = (insn >> 12) & 0xf; | ||
| 639 | int rn = (insn >> 16) & 0xf; | ||
| 640 | |||
| 641 | fnr.dr = insnslot_llret_0arg_rflags(regs->ARM_cpsr, i_fn); | ||
| 642 | regs->uregs[rn] = fnr.r0; | ||
| 643 | regs->uregs[rd] = fnr.r1; | ||
| 644 | } | ||
| 645 | |||
| 646 | static void __kprobes emulate_mcrr(struct kprobe *p, struct pt_regs *regs) | ||
| 647 | { | ||
| 648 | insn_2arg_fn_t *i_fn = (insn_2arg_fn_t *)&p->ainsn.insn[0]; | ||
| 649 | kprobe_opcode_t insn = p->opcode; | ||
| 650 | int rd = (insn >> 12) & 0xf; | ||
| 651 | int rn = (insn >> 16) & 0xf; | ||
| 652 | long rnv = regs->uregs[rn]; | ||
| 653 | long rdv = regs->uregs[rd]; | ||
| 654 | |||
| 655 | insnslot_2arg_rflags(rnv, rdv, regs->ARM_cpsr, i_fn); | ||
| 656 | } | ||
| 657 | |||
| 658 | static void __kprobes emulate_sat(struct kprobe *p, struct pt_regs *regs) | 624 | static void __kprobes emulate_sat(struct kprobe *p, struct pt_regs *regs) |
| 659 | { | 625 | { |
| 660 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | 626 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; |
| @@ -688,32 +654,32 @@ static void __kprobes emulate_none(struct kprobe *p, struct pt_regs *regs) | |||
| 688 | insnslot_0arg_rflags(regs->ARM_cpsr, i_fn); | 654 | insnslot_0arg_rflags(regs->ARM_cpsr, i_fn); |
| 689 | } | 655 | } |
| 690 | 656 | ||
| 691 | static void __kprobes emulate_rd12(struct kprobe *p, struct pt_regs *regs) | 657 | static void __kprobes emulate_nop(struct kprobe *p, struct pt_regs *regs) |
| 692 | { | 658 | { |
| 693 | insn_0arg_fn_t *i_fn = (insn_0arg_fn_t *)&p->ainsn.insn[0]; | ||
| 694 | kprobe_opcode_t insn = p->opcode; | ||
| 695 | int rd = (insn >> 12) & 0xf; | ||
| 696 | |||
| 697 | regs->uregs[rd] = insnslot_0arg_rflags(regs->ARM_cpsr, i_fn); | ||
| 698 | } | 659 | } |
| 699 | 660 | ||
| 700 | static void __kprobes emulate_ird12(struct kprobe *p, struct pt_regs *regs) | 661 | static void __kprobes |
| 662 | emulate_rd12_modify(struct kprobe *p, struct pt_regs *regs) | ||
| 701 | { | 663 | { |
| 702 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | 664 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; |
| 703 | kprobe_opcode_t insn = p->opcode; | 665 | kprobe_opcode_t insn = p->opcode; |
| 704 | int ird = (insn >> 12) & 0xf; | 666 | int rd = (insn >> 12) & 0xf; |
| 667 | long rdv = regs->uregs[rd]; | ||
| 705 | 668 | ||
| 706 | insnslot_1arg_rflags(regs->uregs[ird], regs->ARM_cpsr, i_fn); | 669 | regs->uregs[rd] = insnslot_1arg_rflags(rdv, regs->ARM_cpsr, i_fn); |
| 707 | } | 670 | } |
| 708 | 671 | ||
| 709 | static void __kprobes emulate_rn16(struct kprobe *p, struct pt_regs *regs) | 672 | static void __kprobes |
| 673 | emulate_rd12rn0_modify(struct kprobe *p, struct pt_regs *regs) | ||
| 710 | { | 674 | { |
| 711 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | 675 | insn_2arg_fn_t *i_fn = (insn_2arg_fn_t *)&p->ainsn.insn[0]; |
| 712 | kprobe_opcode_t insn = p->opcode; | 676 | kprobe_opcode_t insn = p->opcode; |
| 713 | int rn = (insn >> 16) & 0xf; | 677 | int rd = (insn >> 12) & 0xf; |
| 678 | int rn = insn & 0xf; | ||
| 679 | long rdv = regs->uregs[rd]; | ||
| 714 | long rnv = regs->uregs[rn]; | 680 | long rnv = regs->uregs[rn]; |
| 715 | 681 | ||
| 716 | insnslot_1arg_rflags(rnv, regs->ARM_cpsr, i_fn); | 682 | regs->uregs[rd] = insnslot_2arg_rflags(rdv, rnv, regs->ARM_cpsr, i_fn); |
| 717 | } | 683 | } |
| 718 | 684 | ||
| 719 | static void __kprobes emulate_rd12rm0(struct kprobe *p, struct pt_regs *regs) | 685 | static void __kprobes emulate_rd12rm0(struct kprobe *p, struct pt_regs *regs) |
| @@ -819,6 +785,17 @@ emulate_alu_imm_rwflags(struct kprobe *p, struct pt_regs *regs) | |||
| 819 | } | 785 | } |
| 820 | 786 | ||
| 821 | static void __kprobes | 787 | static void __kprobes |
| 788 | emulate_alu_tests_imm(struct kprobe *p, struct pt_regs *regs) | ||
| 789 | { | ||
| 790 | insn_1arg_fn_t *i_fn = (insn_1arg_fn_t *)&p->ainsn.insn[0]; | ||
| 791 | kprobe_opcode_t insn = p->opcode; | ||
| 792 | int rn = (insn >> 16) & 0xf; | ||
| 793 | long rnv = (rn == 15) ? (long)p->addr + 8 : regs->uregs[rn]; | ||
| 794 | |||
| 795 | insnslot_1arg_rwflags(rnv, ®s->ARM_cpsr, i_fn); | ||
| 796 | } | ||
| 797 | |||
| 798 | static void __kprobes | ||
| 822 | emulate_alu_rflags(struct kprobe *p, struct pt_regs *regs) | 799 | emulate_alu_rflags(struct kprobe *p, struct pt_regs *regs) |
| 823 | { | 800 | { |
| 824 | insn_3arg_fn_t *i_fn = (insn_3arg_fn_t *)&p->ainsn.insn[0]; | 801 | insn_3arg_fn_t *i_fn = (insn_3arg_fn_t *)&p->ainsn.insn[0]; |
| @@ -854,14 +831,34 @@ emulate_alu_rwflags(struct kprobe *p, struct pt_regs *regs) | |||
| 854 | insnslot_3arg_rwflags(rnv, rmv, rsv, ®s->ARM_cpsr, i_fn); | 831 | insnslot_3arg_rwflags(rnv, rmv, rsv, ®s->ARM_cpsr, i_fn); |
| 855 | } | 832 | } |
| 856 | 833 | ||
| 834 | static void __kprobes | ||
| 835 | emulate_alu_tests(struct kprobe *p, struct pt_regs *regs) | ||
| 836 | { | ||
| 837 | insn_3arg_fn_t *i_fn = (insn_3arg_fn_t *)&p->ainsn.insn[0]; | ||
| 838 | kprobe_opcode_t insn = p->opcode; | ||
| 839 | long ppc = (long)p->addr + 8; | ||
| 840 | int rn = (insn >> 16) & 0xf; | ||
| 841 | int rs = (insn >> 8) & 0xf; /* rs/rsv may be invalid, don't care. */ | ||
| 842 | int rm = insn & 0xf; | ||
| 843 | long rnv = (rn == 15) ? ppc : regs->uregs[rn]; | ||
| 844 | long rmv = (rm == 15) ? ppc : regs->uregs[rm]; | ||
| 845 | long rsv = regs->uregs[rs]; | ||
| 846 | |||
| 847 | insnslot_3arg_rwflags(rnv, rmv, rsv, ®s->ARM_cpsr, i_fn); | ||
| 848 | } | ||
| 849 | |||
| 857 | static enum kprobe_insn __kprobes | 850 | static enum kprobe_insn __kprobes |
| 858 | prep_emulate_ldr_str(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 851 | prep_emulate_ldr_str(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
| 859 | { | 852 | { |
| 860 | int ibit = (insn & (1 << 26)) ? 25 : 22; | 853 | int not_imm = (insn & (1 << 26)) ? (insn & (1 << 25)) |
| 854 | : (~insn & (1 << 22)); | ||
| 855 | |||
| 856 | if (is_writeback(insn) && is_r15(insn, 16)) | ||
| 857 | return INSN_REJECTED; /* Writeback to PC */ | ||
| 861 | 858 | ||
| 862 | insn &= 0xfff00fff; | 859 | insn &= 0xfff00fff; |
| 863 | insn |= 0x00001000; /* Rn = r0, Rd = r1 */ | 860 | insn |= 0x00001000; /* Rn = r0, Rd = r1 */ |
| 864 | if (insn & (1 << ibit)) { | 861 | if (not_imm) { |
| 865 | insn &= ~0xf; | 862 | insn &= ~0xf; |
| 866 | insn |= 2; /* Rm = r2 */ | 863 | insn |= 2; /* Rm = r2 */ |
| 867 | } | 864 | } |
| @@ -871,20 +868,40 @@ prep_emulate_ldr_str(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 871 | } | 868 | } |
| 872 | 869 | ||
| 873 | static enum kprobe_insn __kprobes | 870 | static enum kprobe_insn __kprobes |
| 874 | prep_emulate_rd12rm0(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 871 | prep_emulate_rd12_modify(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
| 875 | { | 872 | { |
| 876 | insn &= 0xffff0ff0; /* Rd = r0, Rm = r0 */ | 873 | if (is_r15(insn, 12)) |
| 874 | return INSN_REJECTED; /* Rd is PC */ | ||
| 875 | |||
| 876 | insn &= 0xffff0fff; /* Rd = r0 */ | ||
| 877 | asi->insn[0] = insn; | 877 | asi->insn[0] = insn; |
| 878 | asi->insn_handler = emulate_rd12rm0; | 878 | asi->insn_handler = emulate_rd12_modify; |
| 879 | return INSN_GOOD; | 879 | return INSN_GOOD; |
| 880 | } | 880 | } |
| 881 | 881 | ||
| 882 | static enum kprobe_insn __kprobes | 882 | static enum kprobe_insn __kprobes |
| 883 | prep_emulate_rd12(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 883 | prep_emulate_rd12rn0_modify(kprobe_opcode_t insn, |
| 884 | struct arch_specific_insn *asi) | ||
| 884 | { | 885 | { |
| 885 | insn &= 0xffff0fff; /* Rd = r0 */ | 886 | if (is_r15(insn, 12)) |
| 887 | return INSN_REJECTED; /* Rd is PC */ | ||
| 888 | |||
| 889 | insn &= 0xffff0ff0; /* Rd = r0 */ | ||
| 890 | insn |= 0x00000001; /* Rn = r1 */ | ||
| 891 | asi->insn[0] = insn; | ||
| 892 | asi->insn_handler = emulate_rd12rn0_modify; | ||
| 893 | return INSN_GOOD; | ||
| 894 | } | ||
| 895 | |||
| 896 | static enum kprobe_insn __kprobes | ||
| 897 | prep_emulate_rd12rm0(kprobe_opcode_t insn, struct arch_specific_insn *asi) | ||
| 898 | { | ||
| 899 | if (is_r15(insn, 12)) | ||
| 900 | return INSN_REJECTED; /* Rd is PC */ | ||
| 901 | |||
| 902 | insn &= 0xffff0ff0; /* Rd = r0, Rm = r0 */ | ||
| 886 | asi->insn[0] = insn; | 903 | asi->insn[0] = insn; |
| 887 | asi->insn_handler = emulate_rd12; | 904 | asi->insn_handler = emulate_rd12rm0; |
| 888 | return INSN_GOOD; | 905 | return INSN_GOOD; |
| 889 | } | 906 | } |
| 890 | 907 | ||
| @@ -892,6 +909,9 @@ static enum kprobe_insn __kprobes | |||
| 892 | prep_emulate_rd12rn16rm0_wflags(kprobe_opcode_t insn, | 909 | prep_emulate_rd12rn16rm0_wflags(kprobe_opcode_t insn, |
| 893 | struct arch_specific_insn *asi) | 910 | struct arch_specific_insn *asi) |
| 894 | { | 911 | { |
| 912 | if (is_r15(insn, 12)) | ||
| 913 | return INSN_REJECTED; /* Rd is PC */ | ||
| 914 | |||
| 895 | insn &= 0xfff00ff0; /* Rd = r0, Rn = r0 */ | 915 | insn &= 0xfff00ff0; /* Rd = r0, Rn = r0 */ |
| 896 | insn |= 0x00000001; /* Rm = r1 */ | 916 | insn |= 0x00000001; /* Rm = r1 */ |
| 897 | asi->insn[0] = insn; | 917 | asi->insn[0] = insn; |
| @@ -903,6 +923,9 @@ static enum kprobe_insn __kprobes | |||
| 903 | prep_emulate_rd16rs8rm0_wflags(kprobe_opcode_t insn, | 923 | prep_emulate_rd16rs8rm0_wflags(kprobe_opcode_t insn, |
| 904 | struct arch_specific_insn *asi) | 924 | struct arch_specific_insn *asi) |
| 905 | { | 925 | { |
| 926 | if (is_r15(insn, 16)) | ||
| 927 | return INSN_REJECTED; /* Rd is PC */ | ||
| 928 | |||
| 906 | insn &= 0xfff0f0f0; /* Rd = r0, Rs = r0 */ | 929 | insn &= 0xfff0f0f0; /* Rd = r0, Rs = r0 */ |
| 907 | insn |= 0x00000001; /* Rm = r1 */ | 930 | insn |= 0x00000001; /* Rm = r1 */ |
| 908 | asi->insn[0] = insn; | 931 | asi->insn[0] = insn; |
| @@ -914,6 +937,9 @@ static enum kprobe_insn __kprobes | |||
| 914 | prep_emulate_rd16rn12rs8rm0_wflags(kprobe_opcode_t insn, | 937 | prep_emulate_rd16rn12rs8rm0_wflags(kprobe_opcode_t insn, |
| 915 | struct arch_specific_insn *asi) | 938 | struct arch_specific_insn *asi) |
| 916 | { | 939 | { |
| 940 | if (is_r15(insn, 16)) | ||
| 941 | return INSN_REJECTED; /* Rd is PC */ | ||
| 942 | |||
| 917 | insn &= 0xfff000f0; /* Rd = r0, Rn = r0 */ | 943 | insn &= 0xfff000f0; /* Rd = r0, Rn = r0 */ |
| 918 | insn |= 0x00000102; /* Rs = r1, Rm = r2 */ | 944 | insn |= 0x00000102; /* Rs = r1, Rm = r2 */ |
| 919 | asi->insn[0] = insn; | 945 | asi->insn[0] = insn; |
| @@ -925,6 +951,9 @@ static enum kprobe_insn __kprobes | |||
| 925 | prep_emulate_rdhi16rdlo12rs8rm0_wflags(kprobe_opcode_t insn, | 951 | prep_emulate_rdhi16rdlo12rs8rm0_wflags(kprobe_opcode_t insn, |
| 926 | struct arch_specific_insn *asi) | 952 | struct arch_specific_insn *asi) |
| 927 | { | 953 | { |
| 954 | if (is_r15(insn, 16) || is_r15(insn, 12)) | ||
| 955 | return INSN_REJECTED; /* RdHi or RdLo is PC */ | ||
| 956 | |||
| 928 | insn &= 0xfff000f0; /* RdHi = r0, RdLo = r1 */ | 957 | insn &= 0xfff000f0; /* RdHi = r0, RdLo = r1 */ |
| 929 | insn |= 0x00001203; /* Rs = r2, Rm = r3 */ | 958 | insn |= 0x00001203; /* Rs = r2, Rm = r3 */ |
| 930 | asi->insn[0] = insn; | 959 | asi->insn[0] = insn; |
| @@ -945,20 +974,13 @@ prep_emulate_rdhi16rdlo12rs8rm0_wflags(kprobe_opcode_t insn, | |||
| 945 | static enum kprobe_insn __kprobes | 974 | static enum kprobe_insn __kprobes |
| 946 | space_1111(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 975 | space_1111(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
| 947 | { | 976 | { |
| 948 | /* CPS mmod == 1 : 1111 0001 0000 xx10 xxxx xxxx xx0x xxxx */ | 977 | /* memory hint : 1111 0100 x001 xxxx xxxx xxxx xxxx xxxx : */ |
| 949 | /* RFE : 1111 100x x0x1 xxxx xxxx 1010 xxxx xxxx */ | 978 | /* PLDI : 1111 0100 x101 xxxx xxxx xxxx xxxx xxxx : */ |
| 950 | /* SRS : 1111 100x x1x0 1101 xxxx 0101 xxxx xxxx */ | 979 | /* PLDW : 1111 0101 x001 xxxx xxxx xxxx xxxx xxxx : */ |
| 951 | if ((insn & 0xfff30020) == 0xf1020000 || | 980 | /* PLD : 1111 0101 x101 xxxx xxxx xxxx xxxx xxxx : */ |
| 952 | (insn & 0xfe500f00) == 0xf8100a00 || | 981 | if ((insn & 0xfe300000) == 0xf4100000) { |
| 953 | (insn & 0xfe5f0f00) == 0xf84d0500) | 982 | asi->insn_handler = emulate_nop; |
| 954 | return INSN_REJECTED; | 983 | return INSN_GOOD_NO_SLOT; |
| 955 | |||
| 956 | /* PLD : 1111 01x1 x101 xxxx xxxx xxxx xxxx xxxx : */ | ||
| 957 | if ((insn & 0xfd700000) == 0xf4500000) { | ||
| 958 | insn &= 0xfff0ffff; /* Rn = r0 */ | ||
| 959 | asi->insn[0] = insn; | ||
| 960 | asi->insn_handler = emulate_rn16; | ||
| 961 | return INSN_GOOD; | ||
| 962 | } | 984 | } |
| 963 | 985 | ||
| 964 | /* BLX(1) : 1111 101x xxxx xxxx xxxx xxxx xxxx xxxx : */ | 986 | /* BLX(1) : 1111 101x xxxx xxxx xxxx xxxx xxxx xxxx : */ |
| @@ -967,41 +989,22 @@ space_1111(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 967 | return INSN_GOOD_NO_SLOT; | 989 | return INSN_GOOD_NO_SLOT; |
| 968 | } | 990 | } |
| 969 | 991 | ||
| 970 | /* SETEND : 1111 0001 0000 0001 xxxx xxxx 0000 xxxx */ | 992 | /* CPS : 1111 0001 0000 xxx0 xxxx xxxx xx0x xxxx */ |
| 971 | /* CDP2 : 1111 1110 xxxx xxxx xxxx xxxx xxx0 xxxx */ | 993 | /* SETEND: 1111 0001 0000 0001 xxxx xxxx 0000 xxxx */ |
| 972 | if ((insn & 0xffff00f0) == 0xf1010000 || | ||
| 973 | (insn & 0xff000010) == 0xfe000000) { | ||
| 974 | asi->insn[0] = insn; | ||
| 975 | asi->insn_handler = emulate_none; | ||
| 976 | return INSN_GOOD; | ||
| 977 | } | ||
| 978 | 994 | ||
| 995 | /* SRS : 1111 100x x1x0 xxxx xxxx xxxx xxxx xxxx */ | ||
| 996 | /* RFE : 1111 100x x0x1 xxxx xxxx xxxx xxxx xxxx */ | ||
| 997 | |||
| 998 | /* Coprocessor instructions... */ | ||
| 979 | /* MCRR2 : 1111 1100 0100 xxxx xxxx xxxx xxxx xxxx : (Rd != Rn) */ | 999 | /* MCRR2 : 1111 1100 0100 xxxx xxxx xxxx xxxx xxxx : (Rd != Rn) */ |
| 980 | /* MRRC2 : 1111 1100 0101 xxxx xxxx xxxx xxxx xxxx : (Rd != Rn) */ | 1000 | /* MRRC2 : 1111 1100 0101 xxxx xxxx xxxx xxxx xxxx : (Rd != Rn) */ |
| 981 | if ((insn & 0xffe00000) == 0xfc400000) { | 1001 | /* LDC2 : 1111 110x xxx1 xxxx xxxx xxxx xxxx xxxx */ |
| 982 | insn &= 0xfff00fff; /* Rn = r0 */ | 1002 | /* STC2 : 1111 110x xxx0 xxxx xxxx xxxx xxxx xxxx */ |
| 983 | insn |= 0x00001000; /* Rd = r1 */ | 1003 | /* CDP2 : 1111 1110 xxxx xxxx xxxx xxxx xxx0 xxxx */ |
| 984 | asi->insn[0] = insn; | 1004 | /* MCR2 : 1111 1110 xxx0 xxxx xxxx xxxx xxx1 xxxx */ |
| 985 | asi->insn_handler = | 1005 | /* MRC2 : 1111 1110 xxx1 xxxx xxxx xxxx xxx1 xxxx */ |
| 986 | (insn & (1 << 20)) ? emulate_mrrc : emulate_mcrr; | ||
| 987 | return INSN_GOOD; | ||
| 988 | } | ||
| 989 | 1006 | ||
| 990 | /* LDC2 : 1111 110x xxx1 xxxx xxxx xxxx xxxx xxxx */ | 1007 | return INSN_REJECTED; |
| 991 | /* STC2 : 1111 110x xxx0 xxxx xxxx xxxx xxxx xxxx */ | ||
| 992 | if ((insn & 0xfe000000) == 0xfc000000) { | ||
| 993 | insn &= 0xfff0ffff; /* Rn = r0 */ | ||
| 994 | asi->insn[0] = insn; | ||
| 995 | asi->insn_handler = emulate_ldcstc; | ||
| 996 | return INSN_GOOD; | ||
| 997 | } | ||
| 998 | |||
| 999 | /* MCR2 : 1111 1110 xxx0 xxxx xxxx xxxx xxx1 xxxx */ | ||
| 1000 | /* MRC2 : 1111 1110 xxx1 xxxx xxxx xxxx xxx1 xxxx */ | ||
| 1001 | insn &= 0xffff0fff; /* Rd = r0 */ | ||
| 1002 | asi->insn[0] = insn; | ||
| 1003 | asi->insn_handler = (insn & (1 << 20)) ? emulate_rd12 : emulate_ird12; | ||
| 1004 | return INSN_GOOD; | ||
| 1005 | } | 1008 | } |
| 1006 | 1009 | ||
| 1007 | static enum kprobe_insn __kprobes | 1010 | static enum kprobe_insn __kprobes |
| @@ -1010,19 +1013,18 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1010 | /* cccc 0001 0xx0 xxxx xxxx xxxx xxxx xxx0 xxxx */ | 1013 | /* cccc 0001 0xx0 xxxx xxxx xxxx xxxx xxx0 xxxx */ |
| 1011 | if ((insn & 0x0f900010) == 0x01000000) { | 1014 | if ((insn & 0x0f900010) == 0x01000000) { |
| 1012 | 1015 | ||
| 1013 | /* BXJ : cccc 0001 0010 xxxx xxxx xxxx 0010 xxxx */ | 1016 | /* MRS cpsr : cccc 0001 0000 xxxx xxxx xxxx 0000 xxxx */ |
| 1014 | /* MSR : cccc 0001 0x10 xxxx xxxx xxxx 0000 xxxx */ | 1017 | if ((insn & 0x0ff000f0) == 0x01000000) { |
| 1015 | if ((insn & 0x0ff000f0) == 0x01200020 || | 1018 | if (is_r15(insn, 12)) |
| 1016 | (insn & 0x0fb000f0) == 0x01200000) | 1019 | return INSN_REJECTED; /* Rd is PC */ |
| 1017 | return INSN_REJECTED; | 1020 | asi->insn_handler = simulate_mrs; |
| 1018 | 1021 | return INSN_GOOD_NO_SLOT; | |
| 1019 | /* MRS : cccc 0001 0x00 xxxx xxxx xxxx 0000 xxxx */ | 1022 | } |
| 1020 | if ((insn & 0x0fb00010) == 0x01000000) | ||
| 1021 | return prep_emulate_rd12(insn, asi); | ||
| 1022 | 1023 | ||
| 1023 | /* SMLALxy : cccc 0001 0100 xxxx xxxx xxxx 1xx0 xxxx */ | 1024 | /* SMLALxy : cccc 0001 0100 xxxx xxxx xxxx 1xx0 xxxx */ |
| 1024 | if ((insn & 0x0ff00090) == 0x01400080) | 1025 | if ((insn & 0x0ff00090) == 0x01400080) |
| 1025 | return prep_emulate_rdhi16rdlo12rs8rm0_wflags(insn, asi); | 1026 | return prep_emulate_rdhi16rdlo12rs8rm0_wflags(insn, |
| 1027 | asi); | ||
| 1026 | 1028 | ||
| 1027 | /* SMULWy : cccc 0001 0010 xxxx xxxx xxxx 1x10 xxxx */ | 1029 | /* SMULWy : cccc 0001 0010 xxxx xxxx xxxx 1x10 xxxx */ |
| 1028 | /* SMULxy : cccc 0001 0110 xxxx xxxx xxxx 1xx0 xxxx */ | 1030 | /* SMULxy : cccc 0001 0110 xxxx xxxx xxxx 1xx0 xxxx */ |
| @@ -1031,24 +1033,29 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1031 | return prep_emulate_rd16rs8rm0_wflags(insn, asi); | 1033 | return prep_emulate_rd16rs8rm0_wflags(insn, asi); |
| 1032 | 1034 | ||
| 1033 | /* SMLAxy : cccc 0001 0000 xxxx xxxx xxxx 1xx0 xxxx : Q */ | 1035 | /* SMLAxy : cccc 0001 0000 xxxx xxxx xxxx 1xx0 xxxx : Q */ |
| 1034 | /* SMLAWy : cccc 0001 0010 xxxx xxxx xxxx 0x00 xxxx : Q */ | 1036 | /* SMLAWy : cccc 0001 0010 xxxx xxxx xxxx 1x00 xxxx : Q */ |
| 1035 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); | 1037 | if ((insn & 0x0ff00090) == 0x01000080 || |
| 1038 | (insn & 0x0ff000b0) == 0x01200080) | ||
| 1039 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); | ||
| 1040 | |||
| 1041 | /* BXJ : cccc 0001 0010 xxxx xxxx xxxx 0010 xxxx */ | ||
| 1042 | /* MSR : cccc 0001 0x10 xxxx xxxx xxxx 0000 xxxx */ | ||
| 1043 | /* MRS spsr : cccc 0001 0100 xxxx xxxx xxxx 0000 xxxx */ | ||
| 1036 | 1044 | ||
| 1045 | /* Other instruction encodings aren't yet defined */ | ||
| 1046 | return INSN_REJECTED; | ||
| 1037 | } | 1047 | } |
| 1038 | 1048 | ||
| 1039 | /* cccc 0001 0xx0 xxxx xxxx xxxx xxxx 0xx1 xxxx */ | 1049 | /* cccc 0001 0xx0 xxxx xxxx xxxx xxxx 0xx1 xxxx */ |
| 1040 | else if ((insn & 0x0f900090) == 0x01000010) { | 1050 | else if ((insn & 0x0f900090) == 0x01000010) { |
| 1041 | 1051 | ||
| 1042 | /* BKPT : 1110 0001 0010 xxxx xxxx xxxx 0111 xxxx */ | ||
| 1043 | if ((insn & 0xfff000f0) == 0xe1200070) | ||
| 1044 | return INSN_REJECTED; | ||
| 1045 | |||
| 1046 | /* BLX(2) : cccc 0001 0010 xxxx xxxx xxxx 0011 xxxx */ | 1052 | /* BLX(2) : cccc 0001 0010 xxxx xxxx xxxx 0011 xxxx */ |
| 1047 | /* BX : cccc 0001 0010 xxxx xxxx xxxx 0001 xxxx */ | 1053 | /* BX : cccc 0001 0010 xxxx xxxx xxxx 0001 xxxx */ |
| 1048 | if ((insn & 0x0ff000d0) == 0x01200010) { | 1054 | if ((insn & 0x0ff000d0) == 0x01200010) { |
| 1049 | asi->insn[0] = truecc_insn(insn); | 1055 | if ((insn & 0x0ff000ff) == 0x0120003f) |
| 1056 | return INSN_REJECTED; /* BLX pc */ | ||
| 1050 | asi->insn_handler = simulate_blx2bx; | 1057 | asi->insn_handler = simulate_blx2bx; |
| 1051 | return INSN_GOOD; | 1058 | return INSN_GOOD_NO_SLOT; |
| 1052 | } | 1059 | } |
| 1053 | 1060 | ||
| 1054 | /* CLZ : cccc 0001 0110 xxxx xxxx xxxx 0001 xxxx */ | 1061 | /* CLZ : cccc 0001 0110 xxxx xxxx xxxx 0001 xxxx */ |
| @@ -1059,17 +1066,27 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1059 | /* QSUB : cccc 0001 0010 xxxx xxxx xxxx 0101 xxxx :Q */ | 1066 | /* QSUB : cccc 0001 0010 xxxx xxxx xxxx 0101 xxxx :Q */ |
| 1060 | /* QDADD : cccc 0001 0100 xxxx xxxx xxxx 0101 xxxx :Q */ | 1067 | /* QDADD : cccc 0001 0100 xxxx xxxx xxxx 0101 xxxx :Q */ |
| 1061 | /* QDSUB : cccc 0001 0110 xxxx xxxx xxxx 0101 xxxx :Q */ | 1068 | /* QDSUB : cccc 0001 0110 xxxx xxxx xxxx 0101 xxxx :Q */ |
| 1062 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | 1069 | if ((insn & 0x0f9000f0) == 0x01000050) |
| 1070 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | ||
| 1071 | |||
| 1072 | /* BKPT : 1110 0001 0010 xxxx xxxx xxxx 0111 xxxx */ | ||
| 1073 | /* SMC : cccc 0001 0110 xxxx xxxx xxxx 0111 xxxx */ | ||
| 1074 | |||
| 1075 | /* Other instruction encodings aren't yet defined */ | ||
| 1076 | return INSN_REJECTED; | ||
| 1063 | } | 1077 | } |
| 1064 | 1078 | ||
| 1065 | /* cccc 0000 xxxx xxxx xxxx xxxx xxxx 1001 xxxx */ | 1079 | /* cccc 0000 xxxx xxxx xxxx xxxx xxxx 1001 xxxx */ |
| 1066 | else if ((insn & 0x0f000090) == 0x00000090) { | 1080 | else if ((insn & 0x0f0000f0) == 0x00000090) { |
| 1067 | 1081 | ||
| 1068 | /* MUL : cccc 0000 0000 xxxx xxxx xxxx 1001 xxxx : */ | 1082 | /* MUL : cccc 0000 0000 xxxx xxxx xxxx 1001 xxxx : */ |
| 1069 | /* MULS : cccc 0000 0001 xxxx xxxx xxxx 1001 xxxx :cc */ | 1083 | /* MULS : cccc 0000 0001 xxxx xxxx xxxx 1001 xxxx :cc */ |
| 1070 | /* MLA : cccc 0000 0010 xxxx xxxx xxxx 1001 xxxx : */ | 1084 | /* MLA : cccc 0000 0010 xxxx xxxx xxxx 1001 xxxx : */ |
| 1071 | /* MLAS : cccc 0000 0011 xxxx xxxx xxxx 1001 xxxx :cc */ | 1085 | /* MLAS : cccc 0000 0011 xxxx xxxx xxxx 1001 xxxx :cc */ |
| 1072 | /* UMAAL : cccc 0000 0100 xxxx xxxx xxxx 1001 xxxx : */ | 1086 | /* UMAAL : cccc 0000 0100 xxxx xxxx xxxx 1001 xxxx : */ |
| 1087 | /* undef : cccc 0000 0101 xxxx xxxx xxxx 1001 xxxx : */ | ||
| 1088 | /* MLS : cccc 0000 0110 xxxx xxxx xxxx 1001 xxxx : */ | ||
| 1089 | /* undef : cccc 0000 0111 xxxx xxxx xxxx 1001 xxxx : */ | ||
| 1073 | /* UMULL : cccc 0000 1000 xxxx xxxx xxxx 1001 xxxx : */ | 1090 | /* UMULL : cccc 0000 1000 xxxx xxxx xxxx 1001 xxxx : */ |
| 1074 | /* UMULLS : cccc 0000 1001 xxxx xxxx xxxx 1001 xxxx :cc */ | 1091 | /* UMULLS : cccc 0000 1001 xxxx xxxx xxxx 1001 xxxx :cc */ |
| 1075 | /* UMLAL : cccc 0000 1010 xxxx xxxx xxxx 1001 xxxx : */ | 1092 | /* UMLAL : cccc 0000 1010 xxxx xxxx xxxx 1001 xxxx : */ |
| @@ -1078,13 +1095,15 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1078 | /* SMULLS : cccc 0000 1101 xxxx xxxx xxxx 1001 xxxx :cc */ | 1095 | /* SMULLS : cccc 0000 1101 xxxx xxxx xxxx 1001 xxxx :cc */ |
| 1079 | /* SMLAL : cccc 0000 1110 xxxx xxxx xxxx 1001 xxxx : */ | 1096 | /* SMLAL : cccc 0000 1110 xxxx xxxx xxxx 1001 xxxx : */ |
| 1080 | /* SMLALS : cccc 0000 1111 xxxx xxxx xxxx 1001 xxxx :cc */ | 1097 | /* SMLALS : cccc 0000 1111 xxxx xxxx xxxx 1001 xxxx :cc */ |
| 1081 | if ((insn & 0x0fe000f0) == 0x00000090) { | 1098 | if ((insn & 0x00d00000) == 0x00500000) |
| 1082 | return prep_emulate_rd16rs8rm0_wflags(insn, asi); | 1099 | return INSN_REJECTED; |
| 1083 | } else if ((insn & 0x0fe000f0) == 0x00200090) { | 1100 | else if ((insn & 0x00e00000) == 0x00000000) |
| 1084 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); | 1101 | return prep_emulate_rd16rs8rm0_wflags(insn, asi); |
| 1085 | } else { | 1102 | else if ((insn & 0x00a00000) == 0x00200000) |
| 1086 | return prep_emulate_rdhi16rdlo12rs8rm0_wflags(insn, asi); | 1103 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); |
| 1087 | } | 1104 | else |
| 1105 | return prep_emulate_rdhi16rdlo12rs8rm0_wflags(insn, | ||
| 1106 | asi); | ||
| 1088 | } | 1107 | } |
| 1089 | 1108 | ||
| 1090 | /* cccc 000x xxxx xxxx xxxx xxxx xxxx 1xx1 xxxx */ | 1109 | /* cccc 000x xxxx xxxx xxxx xxxx xxxx 1xx1 xxxx */ |
| @@ -1092,23 +1111,45 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1092 | 1111 | ||
| 1093 | /* SWP : cccc 0001 0000 xxxx xxxx xxxx 1001 xxxx */ | 1112 | /* SWP : cccc 0001 0000 xxxx xxxx xxxx 1001 xxxx */ |
| 1094 | /* SWPB : cccc 0001 0100 xxxx xxxx xxxx 1001 xxxx */ | 1113 | /* SWPB : cccc 0001 0100 xxxx xxxx xxxx 1001 xxxx */ |
| 1095 | /* LDRD : cccc 000x xxx0 xxxx xxxx xxxx 1101 xxxx */ | 1114 | /* ??? : cccc 0001 0x01 xxxx xxxx xxxx 1001 xxxx */ |
| 1096 | /* STRD : cccc 000x xxx0 xxxx xxxx xxxx 1111 xxxx */ | 1115 | /* ??? : cccc 0001 0x10 xxxx xxxx xxxx 1001 xxxx */ |
| 1116 | /* ??? : cccc 0001 0x11 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1097 | /* STREX : cccc 0001 1000 xxxx xxxx xxxx 1001 xxxx */ | 1117 | /* STREX : cccc 0001 1000 xxxx xxxx xxxx 1001 xxxx */ |
| 1098 | /* LDREX : cccc 0001 1001 xxxx xxxx xxxx 1001 xxxx */ | 1118 | /* LDREX : cccc 0001 1001 xxxx xxxx xxxx 1001 xxxx */ |
| 1119 | /* STREXD: cccc 0001 1010 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1120 | /* LDREXD: cccc 0001 1011 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1121 | /* STREXB: cccc 0001 1100 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1122 | /* LDREXB: cccc 0001 1101 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1123 | /* STREXH: cccc 0001 1110 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1124 | /* LDREXH: cccc 0001 1111 xxxx xxxx xxxx 1001 xxxx */ | ||
| 1125 | |||
| 1126 | /* LDRD : cccc 000x xxx0 xxxx xxxx xxxx 1101 xxxx */ | ||
| 1127 | /* STRD : cccc 000x xxx0 xxxx xxxx xxxx 1111 xxxx */ | ||
| 1099 | /* LDRH : cccc 000x xxx1 xxxx xxxx xxxx 1011 xxxx */ | 1128 | /* LDRH : cccc 000x xxx1 xxxx xxxx xxxx 1011 xxxx */ |
| 1100 | /* STRH : cccc 000x xxx0 xxxx xxxx xxxx 1011 xxxx */ | 1129 | /* STRH : cccc 000x xxx0 xxxx xxxx xxxx 1011 xxxx */ |
| 1101 | /* LDRSB : cccc 000x xxx1 xxxx xxxx xxxx 1101 xxxx */ | 1130 | /* LDRSB : cccc 000x xxx1 xxxx xxxx xxxx 1101 xxxx */ |
| 1102 | /* LDRSH : cccc 000x xxx1 xxxx xxxx xxxx 1111 xxxx */ | 1131 | /* LDRSH : cccc 000x xxx1 xxxx xxxx xxxx 1111 xxxx */ |
| 1103 | if ((insn & 0x0fb000f0) == 0x01000090) { | 1132 | if ((insn & 0x0f0000f0) == 0x01000090) { |
| 1104 | /* SWP/SWPB */ | 1133 | if ((insn & 0x0fb000f0) == 0x01000090) { |
| 1105 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | 1134 | /* SWP/SWPB */ |
| 1135 | return prep_emulate_rd12rn16rm0_wflags(insn, | ||
| 1136 | asi); | ||
| 1137 | } else { | ||
| 1138 | /* STREX/LDREX variants and unallocaed space */ | ||
| 1139 | return INSN_REJECTED; | ||
| 1140 | } | ||
| 1141 | |||
| 1106 | } else if ((insn & 0x0e1000d0) == 0x00000d0) { | 1142 | } else if ((insn & 0x0e1000d0) == 0x00000d0) { |
| 1107 | /* STRD/LDRD */ | 1143 | /* STRD/LDRD */ |
| 1144 | if ((insn & 0x0000e000) == 0x0000e000) | ||
| 1145 | return INSN_REJECTED; /* Rd is LR or PC */ | ||
| 1146 | if (is_writeback(insn) && is_r15(insn, 16)) | ||
| 1147 | return INSN_REJECTED; /* Writeback to PC */ | ||
| 1148 | |||
| 1108 | insn &= 0xfff00fff; | 1149 | insn &= 0xfff00fff; |
| 1109 | insn |= 0x00002000; /* Rn = r0, Rd = r2 */ | 1150 | insn |= 0x00002000; /* Rn = r0, Rd = r2 */ |
| 1110 | if (insn & (1 << 22)) { | 1151 | if (!(insn & (1 << 22))) { |
| 1111 | /* I bit */ | 1152 | /* Register index */ |
| 1112 | insn &= ~0xf; | 1153 | insn &= ~0xf; |
| 1113 | insn |= 1; /* Rm = r1 */ | 1154 | insn |= 1; /* Rm = r1 */ |
| 1114 | } | 1155 | } |
| @@ -1118,6 +1159,9 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1118 | return INSN_GOOD; | 1159 | return INSN_GOOD; |
| 1119 | } | 1160 | } |
| 1120 | 1161 | ||
| 1162 | /* LDRH/STRH/LDRSB/LDRSH */ | ||
| 1163 | if (is_r15(insn, 12)) | ||
| 1164 | return INSN_REJECTED; /* Rd is PC */ | ||
| 1121 | return prep_emulate_ldr_str(insn, asi); | 1165 | return prep_emulate_ldr_str(insn, asi); |
| 1122 | } | 1166 | } |
| 1123 | 1167 | ||
| @@ -1125,7 +1169,7 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1125 | 1169 | ||
| 1126 | /* | 1170 | /* |
| 1127 | * ALU op with S bit and Rd == 15 : | 1171 | * ALU op with S bit and Rd == 15 : |
| 1128 | * cccc 000x xxx1 xxxx 1111 xxxx xxxx xxxx | 1172 | * cccc 000x xxx1 xxxx 1111 xxxx xxxx xxxx |
| 1129 | */ | 1173 | */ |
| 1130 | if ((insn & 0x0e10f000) == 0x0010f000) | 1174 | if ((insn & 0x0e10f000) == 0x0010f000) |
| 1131 | return INSN_REJECTED; | 1175 | return INSN_REJECTED; |
| @@ -1154,22 +1198,61 @@ space_cccc_000x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1154 | insn |= 0x00000200; /* Rs = r2 */ | 1198 | insn |= 0x00000200; /* Rs = r2 */ |
| 1155 | } | 1199 | } |
| 1156 | asi->insn[0] = insn; | 1200 | asi->insn[0] = insn; |
| 1157 | asi->insn_handler = (insn & (1 << 20)) ? /* S-bit */ | 1201 | |
| 1202 | if ((insn & 0x0f900000) == 0x01100000) { | ||
| 1203 | /* | ||
| 1204 | * TST : cccc 0001 0001 xxxx xxxx xxxx xxxx xxxx | ||
| 1205 | * TEQ : cccc 0001 0011 xxxx xxxx xxxx xxxx xxxx | ||
| 1206 | * CMP : cccc 0001 0101 xxxx xxxx xxxx xxxx xxxx | ||
| 1207 | * CMN : cccc 0001 0111 xxxx xxxx xxxx xxxx xxxx | ||
| 1208 | */ | ||
| 1209 | asi->insn_handler = emulate_alu_tests; | ||
| 1210 | } else { | ||
| 1211 | /* ALU ops which write to Rd */ | ||
| 1212 | asi->insn_handler = (insn & (1 << 20)) ? /* S-bit */ | ||
| 1158 | emulate_alu_rwflags : emulate_alu_rflags; | 1213 | emulate_alu_rwflags : emulate_alu_rflags; |
| 1214 | } | ||
| 1159 | return INSN_GOOD; | 1215 | return INSN_GOOD; |
| 1160 | } | 1216 | } |
| 1161 | 1217 | ||
| 1162 | static enum kprobe_insn __kprobes | 1218 | static enum kprobe_insn __kprobes |
| 1163 | space_cccc_001x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 1219 | space_cccc_001x(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
| 1164 | { | 1220 | { |
| 1221 | /* MOVW : cccc 0011 0000 xxxx xxxx xxxx xxxx xxxx */ | ||
| 1222 | /* MOVT : cccc 0011 0100 xxxx xxxx xxxx xxxx xxxx */ | ||
| 1223 | if ((insn & 0x0fb00000) == 0x03000000) | ||
| 1224 | return prep_emulate_rd12_modify(insn, asi); | ||
| 1225 | |||
| 1226 | /* hints : cccc 0011 0010 0000 xxxx xxxx xxxx xxxx */ | ||
| 1227 | if ((insn & 0x0fff0000) == 0x03200000) { | ||
| 1228 | unsigned op2 = insn & 0x000000ff; | ||
| 1229 | if (op2 == 0x01 || op2 == 0x04) { | ||
| 1230 | /* YIELD : cccc 0011 0010 0000 xxxx xxxx 0000 0001 */ | ||
| 1231 | /* SEV : cccc 0011 0010 0000 xxxx xxxx 0000 0100 */ | ||
| 1232 | asi->insn[0] = insn; | ||
| 1233 | asi->insn_handler = emulate_none; | ||
| 1234 | return INSN_GOOD; | ||
| 1235 | } else if (op2 <= 0x03) { | ||
| 1236 | /* NOP : cccc 0011 0010 0000 xxxx xxxx 0000 0000 */ | ||
| 1237 | /* WFE : cccc 0011 0010 0000 xxxx xxxx 0000 0010 */ | ||
| 1238 | /* WFI : cccc 0011 0010 0000 xxxx xxxx 0000 0011 */ | ||
| 1239 | /* | ||
| 1240 | * We make WFE and WFI true NOPs to avoid stalls due | ||
| 1241 | * to missing events whilst processing the probe. | ||
| 1242 | */ | ||
| 1243 | asi->insn_handler = emulate_nop; | ||
| 1244 | return INSN_GOOD_NO_SLOT; | ||
| 1245 | } | ||
| 1246 | /* For DBG and unallocated hints it's safest to reject them */ | ||
| 1247 | return INSN_REJECTED; | ||
| 1248 | } | ||
| 1249 | |||
| 1165 | /* | 1250 | /* |
| 1166 | * MSR : cccc 0011 0x10 xxxx xxxx xxxx xxxx xxxx | 1251 | * MSR : cccc 0011 0x10 xxxx xxxx xxxx xxxx xxxx |
| 1167 | * Undef : cccc 0011 0100 xxxx xxxx xxxx xxxx xxxx | ||
| 1168 | * ALU op with S bit and Rd == 15 : | 1252 | * ALU op with S bit and Rd == 15 : |
| 1169 | * cccc 001x xxx1 xxxx 1111 xxxx xxxx xxxx | 1253 | * cccc 001x xxx1 xxxx 1111 xxxx xxxx xxxx |
| 1170 | */ | 1254 | */ |
| 1171 | if ((insn & 0x0fb00000) == 0x03200000 || /* MSR */ | 1255 | if ((insn & 0x0fb00000) == 0x03200000 || /* MSR */ |
| 1172 | (insn & 0x0ff00000) == 0x03400000 || /* Undef */ | ||
| 1173 | (insn & 0x0e10f000) == 0x0210f000) /* ALU s-bit, R15 */ | 1256 | (insn & 0x0e10f000) == 0x0210f000) /* ALU s-bit, R15 */ |
| 1174 | return INSN_REJECTED; | 1257 | return INSN_REJECTED; |
| 1175 | 1258 | ||
| @@ -1180,10 +1263,22 @@ space_cccc_001x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1180 | * *S (bit 20) updates condition codes | 1263 | * *S (bit 20) updates condition codes |
| 1181 | * ADC/SBC/RSC reads the C flag | 1264 | * ADC/SBC/RSC reads the C flag |
| 1182 | */ | 1265 | */ |
| 1183 | insn &= 0xffff0fff; /* Rd = r0 */ | 1266 | insn &= 0xfff00fff; /* Rn = r0 and Rd = r0 */ |
| 1184 | asi->insn[0] = insn; | 1267 | asi->insn[0] = insn; |
| 1185 | asi->insn_handler = (insn & (1 << 20)) ? /* S-bit */ | 1268 | |
| 1269 | if ((insn & 0x0f900000) == 0x03100000) { | ||
| 1270 | /* | ||
| 1271 | * TST : cccc 0011 0001 xxxx xxxx xxxx xxxx xxxx | ||
| 1272 | * TEQ : cccc 0011 0011 xxxx xxxx xxxx xxxx xxxx | ||
| 1273 | * CMP : cccc 0011 0101 xxxx xxxx xxxx xxxx xxxx | ||
| 1274 | * CMN : cccc 0011 0111 xxxx xxxx xxxx xxxx xxxx | ||
| 1275 | */ | ||
| 1276 | asi->insn_handler = emulate_alu_tests_imm; | ||
| 1277 | } else { | ||
| 1278 | /* ALU ops which write to Rd */ | ||
| 1279 | asi->insn_handler = (insn & (1 << 20)) ? /* S-bit */ | ||
| 1186 | emulate_alu_imm_rwflags : emulate_alu_imm_rflags; | 1280 | emulate_alu_imm_rwflags : emulate_alu_imm_rflags; |
| 1281 | } | ||
| 1187 | return INSN_GOOD; | 1282 | return INSN_GOOD; |
| 1188 | } | 1283 | } |
| 1189 | 1284 | ||
| @@ -1192,6 +1287,8 @@ space_cccc_0110__1(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1192 | { | 1287 | { |
| 1193 | /* SEL : cccc 0110 1000 xxxx xxxx xxxx 1011 xxxx GE: !!! */ | 1288 | /* SEL : cccc 0110 1000 xxxx xxxx xxxx 1011 xxxx GE: !!! */ |
| 1194 | if ((insn & 0x0ff000f0) == 0x068000b0) { | 1289 | if ((insn & 0x0ff000f0) == 0x068000b0) { |
| 1290 | if (is_r15(insn, 12)) | ||
| 1291 | return INSN_REJECTED; /* Rd is PC */ | ||
| 1195 | insn &= 0xfff00ff0; /* Rd = r0, Rn = r0 */ | 1292 | insn &= 0xfff00ff0; /* Rd = r0, Rn = r0 */ |
| 1196 | insn |= 0x00000001; /* Rm = r1 */ | 1293 | insn |= 0x00000001; /* Rm = r1 */ |
| 1197 | asi->insn[0] = insn; | 1294 | asi->insn[0] = insn; |
| @@ -1205,6 +1302,8 @@ space_cccc_0110__1(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1205 | /* USAT16 : cccc 0110 1110 xxxx xxxx xxxx 0011 xxxx :Q */ | 1302 | /* USAT16 : cccc 0110 1110 xxxx xxxx xxxx 0011 xxxx :Q */ |
| 1206 | if ((insn & 0x0fa00030) == 0x06a00010 || | 1303 | if ((insn & 0x0fa00030) == 0x06a00010 || |
| 1207 | (insn & 0x0fb000f0) == 0x06a00030) { | 1304 | (insn & 0x0fb000f0) == 0x06a00030) { |
| 1305 | if (is_r15(insn, 12)) | ||
| 1306 | return INSN_REJECTED; /* Rd is PC */ | ||
| 1208 | insn &= 0xffff0ff0; /* Rd = r0, Rm = r0 */ | 1307 | insn &= 0xffff0ff0; /* Rd = r0, Rm = r0 */ |
| 1209 | asi->insn[0] = insn; | 1308 | asi->insn[0] = insn; |
| 1210 | asi->insn_handler = emulate_sat; | 1309 | asi->insn_handler = emulate_sat; |
| @@ -1213,57 +1312,101 @@ space_cccc_0110__1(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1213 | 1312 | ||
| 1214 | /* REV : cccc 0110 1011 xxxx xxxx xxxx 0011 xxxx */ | 1313 | /* REV : cccc 0110 1011 xxxx xxxx xxxx 0011 xxxx */ |
| 1215 | /* REV16 : cccc 0110 1011 xxxx xxxx xxxx 1011 xxxx */ | 1314 | /* REV16 : cccc 0110 1011 xxxx xxxx xxxx 1011 xxxx */ |
| 1315 | /* RBIT : cccc 0110 1111 xxxx xxxx xxxx 0011 xxxx */ | ||
| 1216 | /* REVSH : cccc 0110 1111 xxxx xxxx xxxx 1011 xxxx */ | 1316 | /* REVSH : cccc 0110 1111 xxxx xxxx xxxx 1011 xxxx */ |
| 1217 | if ((insn & 0x0ff00070) == 0x06b00030 || | 1317 | if ((insn & 0x0ff00070) == 0x06b00030 || |
| 1218 | (insn & 0x0ff000f0) == 0x06f000b0) | 1318 | (insn & 0x0ff00070) == 0x06f00030) |
| 1219 | return prep_emulate_rd12rm0(insn, asi); | 1319 | return prep_emulate_rd12rm0(insn, asi); |
| 1220 | 1320 | ||
| 1321 | /* ??? : cccc 0110 0000 xxxx xxxx xxxx xxx1 xxxx : */ | ||
| 1221 | /* SADD16 : cccc 0110 0001 xxxx xxxx xxxx 0001 xxxx :GE */ | 1322 | /* SADD16 : cccc 0110 0001 xxxx xxxx xxxx 0001 xxxx :GE */ |
| 1222 | /* SADDSUBX : cccc 0110 0001 xxxx xxxx xxxx 0011 xxxx :GE */ | 1323 | /* SADDSUBX : cccc 0110 0001 xxxx xxxx xxxx 0011 xxxx :GE */ |
| 1223 | /* SSUBADDX : cccc 0110 0001 xxxx xxxx xxxx 0101 xxxx :GE */ | 1324 | /* SSUBADDX : cccc 0110 0001 xxxx xxxx xxxx 0101 xxxx :GE */ |
| 1224 | /* SSUB16 : cccc 0110 0001 xxxx xxxx xxxx 0111 xxxx :GE */ | 1325 | /* SSUB16 : cccc 0110 0001 xxxx xxxx xxxx 0111 xxxx :GE */ |
| 1225 | /* SADD8 : cccc 0110 0001 xxxx xxxx xxxx 1001 xxxx :GE */ | 1326 | /* SADD8 : cccc 0110 0001 xxxx xxxx xxxx 1001 xxxx :GE */ |
| 1327 | /* ??? : cccc 0110 0001 xxxx xxxx xxxx 1011 xxxx : */ | ||
| 1328 | /* ??? : cccc 0110 0001 xxxx xxxx xxxx 1101 xxxx : */ | ||
| 1226 | /* SSUB8 : cccc 0110 0001 xxxx xxxx xxxx 1111 xxxx :GE */ | 1329 | /* SSUB8 : cccc 0110 0001 xxxx xxxx xxxx 1111 xxxx :GE */ |
| 1227 | /* QADD16 : cccc 0110 0010 xxxx xxxx xxxx 0001 xxxx : */ | 1330 | /* QADD16 : cccc 0110 0010 xxxx xxxx xxxx 0001 xxxx : */ |
| 1228 | /* QADDSUBX : cccc 0110 0010 xxxx xxxx xxxx 0011 xxxx : */ | 1331 | /* QADDSUBX : cccc 0110 0010 xxxx xxxx xxxx 0011 xxxx : */ |
| 1229 | /* QSUBADDX : cccc 0110 0010 xxxx xxxx xxxx 0101 xxxx : */ | 1332 | /* QSUBADDX : cccc 0110 0010 xxxx xxxx xxxx 0101 xxxx : */ |
| 1230 | /* QSUB16 : cccc 0110 0010 xxxx xxxx xxxx 0111 xxxx : */ | 1333 | /* QSUB16 : cccc 0110 0010 xxxx xxxx xxxx 0111 xxxx : */ |
| 1231 | /* QADD8 : cccc 0110 0010 xxxx xxxx xxxx 1001 xxxx : */ | 1334 | /* QADD8 : cccc 0110 0010 xxxx xxxx xxxx 1001 xxxx : */ |
| 1335 | /* ??? : cccc 0110 0010 xxxx xxxx xxxx 1011 xxxx : */ | ||
| 1336 | /* ??? : cccc 0110 0010 xxxx xxxx xxxx 1101 xxxx : */ | ||
| 1232 | /* QSUB8 : cccc 0110 0010 xxxx xxxx xxxx 1111 xxxx : */ | 1337 | /* QSUB8 : cccc 0110 0010 xxxx xxxx xxxx 1111 xxxx : */ |
| 1233 | /* SHADD16 : cccc 0110 0011 xxxx xxxx xxxx 0001 xxxx : */ | 1338 | /* SHADD16 : cccc 0110 0011 xxxx xxxx xxxx 0001 xxxx : */ |
| 1234 | /* SHADDSUBX : cccc 0110 0011 xxxx xxxx xxxx 0011 xxxx : */ | 1339 | /* SHADDSUBX : cccc 0110 0011 xxxx xxxx xxxx 0011 xxxx : */ |
| 1235 | /* SHSUBADDX : cccc 0110 0011 xxxx xxxx xxxx 0101 xxxx : */ | 1340 | /* SHSUBADDX : cccc 0110 0011 xxxx xxxx xxxx 0101 xxxx : */ |
| 1236 | /* SHSUB16 : cccc 0110 0011 xxxx xxxx xxxx 0111 xxxx : */ | 1341 | /* SHSUB16 : cccc 0110 0011 xxxx xxxx xxxx 0111 xxxx : */ |
| 1237 | /* SHADD8 : cccc 0110 0011 xxxx xxxx xxxx 1001 xxxx : */ | 1342 | /* SHADD8 : cccc 0110 0011 xxxx xxxx xxxx 1001 xxxx : */ |
| 1343 | /* ??? : cccc 0110 0011 xxxx xxxx xxxx 1011 xxxx : */ | ||
| 1344 | /* ??? : cccc 0110 0011 xxxx xxxx xxxx 1101 xxxx : */ | ||
| 1238 | /* SHSUB8 : cccc 0110 0011 xxxx xxxx xxxx 1111 xxxx : */ | 1345 | /* SHSUB8 : cccc 0110 0011 xxxx xxxx xxxx 1111 xxxx : */ |
| 1346 | /* ??? : cccc 0110 0100 xxxx xxxx xxxx xxx1 xxxx : */ | ||
| 1239 | /* UADD16 : cccc 0110 0101 xxxx xxxx xxxx 0001 xxxx :GE */ | 1347 | /* UADD16 : cccc 0110 0101 xxxx xxxx xxxx 0001 xxxx :GE */ |
| 1240 | /* UADDSUBX : cccc 0110 0101 xxxx xxxx xxxx 0011 xxxx :GE */ | 1348 | /* UADDSUBX : cccc 0110 0101 xxxx xxxx xxxx 0011 xxxx :GE */ |
| 1241 | /* USUBADDX : cccc 0110 0101 xxxx xxxx xxxx 0101 xxxx :GE */ | 1349 | /* USUBADDX : cccc 0110 0101 xxxx xxxx xxxx 0101 xxxx :GE */ |
| 1242 | /* USUB16 : cccc 0110 0101 xxxx xxxx xxxx 0111 xxxx :GE */ | 1350 | /* USUB16 : cccc 0110 0101 xxxx xxxx xxxx 0111 xxxx :GE */ |
| 1243 | /* UADD8 : cccc 0110 0101 xxxx xxxx xxxx 1001 xxxx :GE */ | 1351 | /* UADD8 : cccc 0110 0101 xxxx xxxx xxxx 1001 xxxx :GE */ |
| 1352 | /* ??? : cccc 0110 0101 xxxx xxxx xxxx 1011 xxxx : */ | ||
| 1353 | /* ??? : cccc 0110 0101 xxxx xxxx xxxx 1101 xxxx : */ | ||
| 1244 | /* USUB8 : cccc 0110 0101 xxxx xxxx xxxx 1111 xxxx :GE */ | 1354 | /* USUB8 : cccc 0110 0101 xxxx xxxx xxxx 1111 xxxx :GE */ |
| 1245 | /* UQADD16 : cccc 0110 0110 xxxx xxxx xxxx 0001 xxxx : */ | 1355 | /* UQADD16 : cccc 0110 0110 xxxx xxxx xxxx 0001 xxxx : */ |
| 1246 | /* UQADDSUBX : cccc 0110 0110 xxxx xxxx xxxx 0011 xxxx : */ | 1356 | /* UQADDSUBX : cccc 0110 0110 xxxx xxxx xxxx 0011 xxxx : */ |
| 1247 | /* UQSUBADDX : cccc 0110 0110 xxxx xxxx xxxx 0101 xxxx : */ | 1357 | /* UQSUBADDX : cccc 0110 0110 xxxx xxxx xxxx 0101 xxxx : */ |
| 1248 | /* UQSUB16 : cccc 0110 0110 xxxx xxxx xxxx 0111 xxxx : */ | 1358 | /* UQSUB16 : cccc 0110 0110 xxxx xxxx xxxx 0111 xxxx : */ |
| 1249 | /* UQADD8 : cccc 0110 0110 xxxx xxxx xxxx 1001 xxxx : */ | 1359 | /* UQADD8 : cccc 0110 0110 xxxx xxxx xxxx 1001 xxxx : */ |
| 1360 | /* ??? : cccc 0110 0110 xxxx xxxx xxxx 1011 xxxx : */ | ||
| 1361 | /* ??? : cccc 0110 0110 xxxx xxxx xxxx 1101 xxxx : */ | ||
| 1250 | /* UQSUB8 : cccc 0110 0110 xxxx xxxx xxxx 1111 xxxx : */ | 1362 | /* UQSUB8 : cccc 0110 0110 xxxx xxxx xxxx 1111 xxxx : */ |
| 1251 | /* UHADD16 : cccc 0110 0111 xxxx xxxx xxxx 0001 xxxx : */ | 1363 | /* UHADD16 : cccc 0110 0111 xxxx xxxx xxxx 0001 xxxx : */ |
| 1252 | /* UHADDSUBX : cccc 0110 0111 xxxx xxxx xxxx 0011 xxxx : */ | 1364 | /* UHADDSUBX : cccc 0110 0111 xxxx xxxx xxxx 0011 xxxx : */ |
| 1253 | /* UHSUBADDX : cccc 0110 0111 xxxx xxxx xxxx 0101 xxxx : */ | 1365 | /* UHSUBADDX : cccc 0110 0111 xxxx xxxx xxxx 0101 xxxx : */ |
| 1254 | /* UHSUB16 : cccc 0110 0111 xxxx xxxx xxxx 0111 xxxx : */ | 1366 | /* UHSUB16 : cccc 0110 0111 xxxx xxxx xxxx 0111 xxxx : */ |
| 1255 | /* UHADD8 : cccc 0110 0111 xxxx xxxx xxxx 1001 xxxx : */ | 1367 | /* UHADD8 : cccc 0110 0111 xxxx xxxx xxxx 1001 xxxx : */ |
| 1368 | /* ??? : cccc 0110 0111 xxxx xxxx xxxx 1011 xxxx : */ | ||
| 1369 | /* ??? : cccc 0110 0111 xxxx xxxx xxxx 1101 xxxx : */ | ||
| 1256 | /* UHSUB8 : cccc 0110 0111 xxxx xxxx xxxx 1111 xxxx : */ | 1370 | /* UHSUB8 : cccc 0110 0111 xxxx xxxx xxxx 1111 xxxx : */ |
| 1371 | if ((insn & 0x0f800010) == 0x06000010) { | ||
| 1372 | if ((insn & 0x00300000) == 0x00000000 || | ||
| 1373 | (insn & 0x000000e0) == 0x000000a0 || | ||
| 1374 | (insn & 0x000000e0) == 0x000000c0) | ||
| 1375 | return INSN_REJECTED; /* Unallocated space */ | ||
| 1376 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | ||
| 1377 | } | ||
| 1378 | |||
| 1257 | /* PKHBT : cccc 0110 1000 xxxx xxxx xxxx x001 xxxx : */ | 1379 | /* PKHBT : cccc 0110 1000 xxxx xxxx xxxx x001 xxxx : */ |
| 1258 | /* PKHTB : cccc 0110 1000 xxxx xxxx xxxx x101 xxxx : */ | 1380 | /* PKHTB : cccc 0110 1000 xxxx xxxx xxxx x101 xxxx : */ |
| 1381 | if ((insn & 0x0ff00030) == 0x06800010) | ||
| 1382 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | ||
| 1383 | |||
| 1259 | /* SXTAB16 : cccc 0110 1000 xxxx xxxx xxxx 0111 xxxx : */ | 1384 | /* SXTAB16 : cccc 0110 1000 xxxx xxxx xxxx 0111 xxxx : */ |
| 1260 | /* SXTB : cccc 0110 1010 xxxx xxxx xxxx 0111 xxxx : */ | 1385 | /* SXTB16 : cccc 0110 1000 1111 xxxx xxxx 0111 xxxx : */ |
| 1386 | /* ??? : cccc 0110 1001 xxxx xxxx xxxx 0111 xxxx : */ | ||
| 1261 | /* SXTAB : cccc 0110 1010 xxxx xxxx xxxx 0111 xxxx : */ | 1387 | /* SXTAB : cccc 0110 1010 xxxx xxxx xxxx 0111 xxxx : */ |
| 1388 | /* SXTB : cccc 0110 1010 1111 xxxx xxxx 0111 xxxx : */ | ||
| 1262 | /* SXTAH : cccc 0110 1011 xxxx xxxx xxxx 0111 xxxx : */ | 1389 | /* SXTAH : cccc 0110 1011 xxxx xxxx xxxx 0111 xxxx : */ |
| 1390 | /* SXTH : cccc 0110 1011 1111 xxxx xxxx 0111 xxxx : */ | ||
| 1263 | /* UXTAB16 : cccc 0110 1100 xxxx xxxx xxxx 0111 xxxx : */ | 1391 | /* UXTAB16 : cccc 0110 1100 xxxx xxxx xxxx 0111 xxxx : */ |
| 1392 | /* UXTB16 : cccc 0110 1100 1111 xxxx xxxx 0111 xxxx : */ | ||
| 1393 | /* ??? : cccc 0110 1101 xxxx xxxx xxxx 0111 xxxx : */ | ||
| 1264 | /* UXTAB : cccc 0110 1110 xxxx xxxx xxxx 0111 xxxx : */ | 1394 | /* UXTAB : cccc 0110 1110 xxxx xxxx xxxx 0111 xxxx : */ |
| 1395 | /* UXTB : cccc 0110 1110 1111 xxxx xxxx 0111 xxxx : */ | ||
| 1265 | /* UXTAH : cccc 0110 1111 xxxx xxxx xxxx 0111 xxxx : */ | 1396 | /* UXTAH : cccc 0110 1111 xxxx xxxx xxxx 0111 xxxx : */ |
| 1266 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | 1397 | /* UXTH : cccc 0110 1111 1111 xxxx xxxx 0111 xxxx : */ |
| 1398 | if ((insn & 0x0f8000f0) == 0x06800070) { | ||
| 1399 | if ((insn & 0x00300000) == 0x00100000) | ||
| 1400 | return INSN_REJECTED; /* Unallocated space */ | ||
| 1401 | |||
| 1402 | if ((insn & 0x000f0000) == 0x000f0000) | ||
| 1403 | return prep_emulate_rd12rm0(insn, asi); | ||
| 1404 | else | ||
| 1405 | return prep_emulate_rd12rn16rm0_wflags(insn, asi); | ||
| 1406 | } | ||
| 1407 | |||
| 1408 | /* Other instruction encodings aren't yet defined */ | ||
| 1409 | return INSN_REJECTED; | ||
| 1267 | } | 1410 | } |
| 1268 | 1411 | ||
| 1269 | static enum kprobe_insn __kprobes | 1412 | static enum kprobe_insn __kprobes |
| @@ -1273,29 +1416,49 @@ space_cccc_0111__1(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1273 | if ((insn & 0x0ff000f0) == 0x03f000f0) | 1416 | if ((insn & 0x0ff000f0) == 0x03f000f0) |
| 1274 | return INSN_REJECTED; | 1417 | return INSN_REJECTED; |
| 1275 | 1418 | ||
| 1276 | /* USADA8 : cccc 0111 1000 xxxx xxxx xxxx 0001 xxxx */ | ||
| 1277 | /* USAD8 : cccc 0111 1000 xxxx 1111 xxxx 0001 xxxx */ | ||
| 1278 | if ((insn & 0x0ff000f0) == 0x07800010) | ||
| 1279 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); | ||
| 1280 | |||
| 1281 | /* SMLALD : cccc 0111 0100 xxxx xxxx xxxx 00x1 xxxx */ | 1419 | /* SMLALD : cccc 0111 0100 xxxx xxxx xxxx 00x1 xxxx */ |
| 1282 | /* SMLSLD : cccc 0111 0100 xxxx xxxx xxxx 01x1 xxxx */ | 1420 | /* SMLSLD : cccc 0111 0100 xxxx xxxx xxxx 01x1 xxxx */ |
| 1283 | if ((insn & 0x0ff00090) == 0x07400010) | 1421 | if ((insn & 0x0ff00090) == 0x07400010) |
| 1284 | return prep_emulate_rdhi16rdlo12rs8rm0_wflags(insn, asi); | 1422 | return prep_emulate_rdhi16rdlo12rs8rm0_wflags(insn, asi); |
| 1285 | 1423 | ||
| 1286 | /* SMLAD : cccc 0111 0000 xxxx xxxx xxxx 00x1 xxxx :Q */ | 1424 | /* SMLAD : cccc 0111 0000 xxxx xxxx xxxx 00x1 xxxx :Q */ |
| 1425 | /* SMUAD : cccc 0111 0000 xxxx 1111 xxxx 00x1 xxxx :Q */ | ||
| 1287 | /* SMLSD : cccc 0111 0000 xxxx xxxx xxxx 01x1 xxxx :Q */ | 1426 | /* SMLSD : cccc 0111 0000 xxxx xxxx xxxx 01x1 xxxx :Q */ |
| 1427 | /* SMUSD : cccc 0111 0000 xxxx 1111 xxxx 01x1 xxxx : */ | ||
| 1288 | /* SMMLA : cccc 0111 0101 xxxx xxxx xxxx 00x1 xxxx : */ | 1428 | /* SMMLA : cccc 0111 0101 xxxx xxxx xxxx 00x1 xxxx : */ |
| 1289 | /* SMMLS : cccc 0111 0101 xxxx xxxx xxxx 11x1 xxxx : */ | 1429 | /* SMMUL : cccc 0111 0101 xxxx 1111 xxxx 00x1 xxxx : */ |
| 1430 | /* USADA8 : cccc 0111 1000 xxxx xxxx xxxx 0001 xxxx : */ | ||
| 1431 | /* USAD8 : cccc 0111 1000 xxxx 1111 xxxx 0001 xxxx : */ | ||
| 1290 | if ((insn & 0x0ff00090) == 0x07000010 || | 1432 | if ((insn & 0x0ff00090) == 0x07000010 || |
| 1291 | (insn & 0x0ff000d0) == 0x07500010 || | 1433 | (insn & 0x0ff000d0) == 0x07500010 || |
| 1292 | (insn & 0x0ff000d0) == 0x075000d0) | 1434 | (insn & 0x0ff000f0) == 0x07800010) { |
| 1435 | |||
| 1436 | if ((insn & 0x0000f000) == 0x0000f000) | ||
| 1437 | return prep_emulate_rd16rs8rm0_wflags(insn, asi); | ||
| 1438 | else | ||
| 1439 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); | ||
| 1440 | } | ||
| 1441 | |||
| 1442 | /* SMMLS : cccc 0111 0101 xxxx xxxx xxxx 11x1 xxxx : */ | ||
| 1443 | if ((insn & 0x0ff000d0) == 0x075000d0) | ||
| 1293 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); | 1444 | return prep_emulate_rd16rn12rs8rm0_wflags(insn, asi); |
| 1294 | 1445 | ||
| 1295 | /* SMUSD : cccc 0111 0000 xxxx xxxx xxxx 01x1 xxxx : */ | 1446 | /* SBFX : cccc 0111 101x xxxx xxxx xxxx x101 xxxx : */ |
| 1296 | /* SMUAD : cccc 0111 0000 xxxx 1111 xxxx 00x1 xxxx :Q */ | 1447 | /* UBFX : cccc 0111 111x xxxx xxxx xxxx x101 xxxx : */ |
| 1297 | /* SMMUL : cccc 0111 0101 xxxx 1111 xxxx 00x1 xxxx : */ | 1448 | if ((insn & 0x0fa00070) == 0x07a00050) |
| 1298 | return prep_emulate_rd16rs8rm0_wflags(insn, asi); | 1449 | return prep_emulate_rd12rm0(insn, asi); |
| 1450 | |||
| 1451 | /* BFI : cccc 0111 110x xxxx xxxx xxxx x001 xxxx : */ | ||
| 1452 | /* BFC : cccc 0111 110x xxxx xxxx xxxx x001 1111 : */ | ||
| 1453 | if ((insn & 0x0fe00070) == 0x07c00010) { | ||
| 1454 | |||
| 1455 | if ((insn & 0x0000000f) == 0x0000000f) | ||
| 1456 | return prep_emulate_rd12_modify(insn, asi); | ||
| 1457 | else | ||
| 1458 | return prep_emulate_rd12rn0_modify(insn, asi); | ||
| 1459 | } | ||
| 1460 | |||
| 1461 | return INSN_REJECTED; | ||
| 1299 | } | 1462 | } |
| 1300 | 1463 | ||
| 1301 | static enum kprobe_insn __kprobes | 1464 | static enum kprobe_insn __kprobes |
| @@ -1309,6 +1472,10 @@ space_cccc_01xx(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1309 | /* STRB : cccc 01xx x1x0 xxxx xxxx xxxx xxxx xxxx */ | 1472 | /* STRB : cccc 01xx x1x0 xxxx xxxx xxxx xxxx xxxx */ |
| 1310 | /* STRBT : cccc 01x0 x110 xxxx xxxx xxxx xxxx xxxx */ | 1473 | /* STRBT : cccc 01x0 x110 xxxx xxxx xxxx xxxx xxxx */ |
| 1311 | /* STRT : cccc 01x0 x010 xxxx xxxx xxxx xxxx xxxx */ | 1474 | /* STRT : cccc 01x0 x010 xxxx xxxx xxxx xxxx xxxx */ |
| 1475 | |||
| 1476 | if ((insn & 0x00500000) == 0x00500000 && is_r15(insn, 12)) | ||
| 1477 | return INSN_REJECTED; /* LDRB into PC */ | ||
| 1478 | |||
| 1312 | return prep_emulate_ldr_str(insn, asi); | 1479 | return prep_emulate_ldr_str(insn, asi); |
| 1313 | } | 1480 | } |
| 1314 | 1481 | ||
| @@ -1323,10 +1490,9 @@ space_cccc_100x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1323 | 1490 | ||
| 1324 | /* LDM(1) : cccc 100x x0x1 xxxx xxxx xxxx xxxx xxxx */ | 1491 | /* LDM(1) : cccc 100x x0x1 xxxx xxxx xxxx xxxx xxxx */ |
| 1325 | /* STM(1) : cccc 100x x0x0 xxxx xxxx xxxx xxxx xxxx */ | 1492 | /* STM(1) : cccc 100x x0x0 xxxx xxxx xxxx xxxx xxxx */ |
| 1326 | asi->insn[0] = truecc_insn(insn); | ||
| 1327 | asi->insn_handler = ((insn & 0x108000) == 0x008000) ? /* STM & R15 */ | 1493 | asi->insn_handler = ((insn & 0x108000) == 0x008000) ? /* STM & R15 */ |
| 1328 | simulate_stm1_pc : simulate_ldm1stm1; | 1494 | simulate_stm1_pc : simulate_ldm1stm1; |
| 1329 | return INSN_GOOD; | 1495 | return INSN_GOOD_NO_SLOT; |
| 1330 | } | 1496 | } |
| 1331 | 1497 | ||
| 1332 | static enum kprobe_insn __kprobes | 1498 | static enum kprobe_insn __kprobes |
| @@ -1334,58 +1500,117 @@ space_cccc_101x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1334 | { | 1500 | { |
| 1335 | /* B : cccc 1010 xxxx xxxx xxxx xxxx xxxx xxxx */ | 1501 | /* B : cccc 1010 xxxx xxxx xxxx xxxx xxxx xxxx */ |
| 1336 | /* BL : cccc 1011 xxxx xxxx xxxx xxxx xxxx xxxx */ | 1502 | /* BL : cccc 1011 xxxx xxxx xxxx xxxx xxxx xxxx */ |
| 1337 | asi->insn[0] = truecc_insn(insn); | ||
| 1338 | asi->insn_handler = simulate_bbl; | 1503 | asi->insn_handler = simulate_bbl; |
| 1339 | return INSN_GOOD; | 1504 | return INSN_GOOD_NO_SLOT; |
| 1340 | } | 1505 | } |
| 1341 | 1506 | ||
| 1342 | static enum kprobe_insn __kprobes | 1507 | static enum kprobe_insn __kprobes |
| 1343 | space_cccc_1100_010x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 1508 | space_cccc_11xx(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
| 1344 | { | 1509 | { |
| 1510 | /* Coprocessor instructions... */ | ||
| 1345 | /* MCRR : cccc 1100 0100 xxxx xxxx xxxx xxxx xxxx : (Rd!=Rn) */ | 1511 | /* MCRR : cccc 1100 0100 xxxx xxxx xxxx xxxx xxxx : (Rd!=Rn) */ |
| 1346 | /* MRRC : cccc 1100 0101 xxxx xxxx xxxx xxxx xxxx : (Rd!=Rn) */ | 1512 | /* MRRC : cccc 1100 0101 xxxx xxxx xxxx xxxx xxxx : (Rd!=Rn) */ |
| 1347 | insn &= 0xfff00fff; | 1513 | /* LDC : cccc 110x xxx1 xxxx xxxx xxxx xxxx xxxx */ |
| 1348 | insn |= 0x00001000; /* Rn = r0, Rd = r1 */ | 1514 | /* STC : cccc 110x xxx0 xxxx xxxx xxxx xxxx xxxx */ |
| 1349 | asi->insn[0] = insn; | 1515 | /* CDP : cccc 1110 xxxx xxxx xxxx xxxx xxx0 xxxx */ |
| 1350 | asi->insn_handler = (insn & (1 << 20)) ? emulate_mrrc : emulate_mcrr; | 1516 | /* MCR : cccc 1110 xxx0 xxxx xxxx xxxx xxx1 xxxx */ |
| 1351 | return INSN_GOOD; | 1517 | /* MRC : cccc 1110 xxx1 xxxx xxxx xxxx xxx1 xxxx */ |
| 1518 | |||
| 1519 | /* SVC : cccc 1111 xxxx xxxx xxxx xxxx xxxx xxxx */ | ||
| 1520 | |||
| 1521 | return INSN_REJECTED; | ||
| 1352 | } | 1522 | } |
| 1353 | 1523 | ||
| 1354 | static enum kprobe_insn __kprobes | 1524 | static unsigned long __kprobes __check_eq(unsigned long cpsr) |
| 1355 | space_cccc_110x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | ||
| 1356 | { | 1525 | { |
| 1357 | /* LDC : cccc 110x xxx1 xxxx xxxx xxxx xxxx xxxx */ | 1526 | return cpsr & PSR_Z_BIT; |
| 1358 | /* STC : cccc 110x xxx0 xxxx xxxx xxxx xxxx xxxx */ | ||
| 1359 | insn &= 0xfff0ffff; /* Rn = r0 */ | ||
| 1360 | asi->insn[0] = insn; | ||
| 1361 | asi->insn_handler = emulate_ldcstc; | ||
| 1362 | return INSN_GOOD; | ||
| 1363 | } | 1527 | } |
| 1364 | 1528 | ||
| 1365 | static enum kprobe_insn __kprobes | 1529 | static unsigned long __kprobes __check_ne(unsigned long cpsr) |
| 1366 | space_cccc_111x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | ||
| 1367 | { | 1530 | { |
| 1368 | /* BKPT : 1110 0001 0010 xxxx xxxx xxxx 0111 xxxx */ | 1531 | return (~cpsr) & PSR_Z_BIT; |
| 1369 | /* SWI : cccc 1111 xxxx xxxx xxxx xxxx xxxx xxxx */ | 1532 | } |
| 1370 | if ((insn & 0xfff000f0) == 0xe1200070 || | ||
| 1371 | (insn & 0x0f000000) == 0x0f000000) | ||
| 1372 | return INSN_REJECTED; | ||
| 1373 | 1533 | ||
| 1374 | /* CDP : cccc 1110 xxxx xxxx xxxx xxxx xxx0 xxxx */ | 1534 | static unsigned long __kprobes __check_cs(unsigned long cpsr) |
| 1375 | if ((insn & 0x0f000010) == 0x0e000000) { | 1535 | { |
| 1376 | asi->insn[0] = insn; | 1536 | return cpsr & PSR_C_BIT; |
| 1377 | asi->insn_handler = emulate_none; | 1537 | } |
| 1378 | return INSN_GOOD; | ||
| 1379 | } | ||
| 1380 | 1538 | ||
| 1381 | /* MCR : cccc 1110 xxx0 xxxx xxxx xxxx xxx1 xxxx */ | 1539 | static unsigned long __kprobes __check_cc(unsigned long cpsr) |
| 1382 | /* MRC : cccc 1110 xxx1 xxxx xxxx xxxx xxx1 xxxx */ | 1540 | { |
| 1383 | insn &= 0xffff0fff; /* Rd = r0 */ | 1541 | return (~cpsr) & PSR_C_BIT; |
| 1384 | asi->insn[0] = insn; | 1542 | } |
| 1385 | asi->insn_handler = (insn & (1 << 20)) ? emulate_rd12 : emulate_ird12; | 1543 | |
| 1386 | return INSN_GOOD; | 1544 | static unsigned long __kprobes __check_mi(unsigned long cpsr) |
| 1545 | { | ||
| 1546 | return cpsr & PSR_N_BIT; | ||
| 1547 | } | ||
| 1548 | |||
| 1549 | static unsigned long __kprobes __check_pl(unsigned long cpsr) | ||
| 1550 | { | ||
| 1551 | return (~cpsr) & PSR_N_BIT; | ||
| 1552 | } | ||
| 1553 | |||
| 1554 | static unsigned long __kprobes __check_vs(unsigned long cpsr) | ||
| 1555 | { | ||
| 1556 | return cpsr & PSR_V_BIT; | ||
| 1557 | } | ||
| 1558 | |||
| 1559 | static unsigned long __kprobes __check_vc(unsigned long cpsr) | ||
| 1560 | { | ||
| 1561 | return (~cpsr) & PSR_V_BIT; | ||
| 1562 | } | ||
| 1563 | |||
| 1564 | static unsigned long __kprobes __check_hi(unsigned long cpsr) | ||
| 1565 | { | ||
| 1566 | cpsr &= ~(cpsr >> 1); /* PSR_C_BIT &= ~PSR_Z_BIT */ | ||
| 1567 | return cpsr & PSR_C_BIT; | ||
| 1387 | } | 1568 | } |
| 1388 | 1569 | ||
| 1570 | static unsigned long __kprobes __check_ls(unsigned long cpsr) | ||
| 1571 | { | ||
| 1572 | cpsr &= ~(cpsr >> 1); /* PSR_C_BIT &= ~PSR_Z_BIT */ | ||
| 1573 | return (~cpsr) & PSR_C_BIT; | ||
| 1574 | } | ||
| 1575 | |||
| 1576 | static unsigned long __kprobes __check_ge(unsigned long cpsr) | ||
| 1577 | { | ||
| 1578 | cpsr ^= (cpsr << 3); /* PSR_N_BIT ^= PSR_V_BIT */ | ||
| 1579 | return (~cpsr) & PSR_N_BIT; | ||
| 1580 | } | ||
| 1581 | |||
| 1582 | static unsigned long __kprobes __check_lt(unsigned long cpsr) | ||
| 1583 | { | ||
| 1584 | cpsr ^= (cpsr << 3); /* PSR_N_BIT ^= PSR_V_BIT */ | ||
| 1585 | return cpsr & PSR_N_BIT; | ||
| 1586 | } | ||
| 1587 | |||
| 1588 | static unsigned long __kprobes __check_gt(unsigned long cpsr) | ||
| 1589 | { | ||
| 1590 | unsigned long temp = cpsr ^ (cpsr << 3); /* PSR_N_BIT ^= PSR_V_BIT */ | ||
| 1591 | temp |= (cpsr << 1); /* PSR_N_BIT |= PSR_Z_BIT */ | ||
| 1592 | return (~temp) & PSR_N_BIT; | ||
| 1593 | } | ||
| 1594 | |||
| 1595 | static unsigned long __kprobes __check_le(unsigned long cpsr) | ||
| 1596 | { | ||
| 1597 | unsigned long temp = cpsr ^ (cpsr << 3); /* PSR_N_BIT ^= PSR_V_BIT */ | ||
| 1598 | temp |= (cpsr << 1); /* PSR_N_BIT |= PSR_Z_BIT */ | ||
| 1599 | return temp & PSR_N_BIT; | ||
| 1600 | } | ||
| 1601 | |||
| 1602 | static unsigned long __kprobes __check_al(unsigned long cpsr) | ||
| 1603 | { | ||
| 1604 | return true; | ||
| 1605 | } | ||
| 1606 | |||
| 1607 | static kprobe_check_cc * const condition_checks[16] = { | ||
| 1608 | &__check_eq, &__check_ne, &__check_cs, &__check_cc, | ||
| 1609 | &__check_mi, &__check_pl, &__check_vs, &__check_vc, | ||
| 1610 | &__check_hi, &__check_ls, &__check_ge, &__check_lt, | ||
| 1611 | &__check_gt, &__check_le, &__check_al, &__check_al | ||
| 1612 | }; | ||
| 1613 | |||
| 1389 | /* Return: | 1614 | /* Return: |
| 1390 | * INSN_REJECTED If instruction is one not allowed to kprobe, | 1615 | * INSN_REJECTED If instruction is one not allowed to kprobe, |
| 1391 | * INSN_GOOD If instruction is supported and uses instruction slot, | 1616 | * INSN_GOOD If instruction is supported and uses instruction slot, |
| @@ -1401,133 +1626,45 @@ space_cccc_111x(kprobe_opcode_t insn, struct arch_specific_insn *asi) | |||
| 1401 | enum kprobe_insn __kprobes | 1626 | enum kprobe_insn __kprobes |
| 1402 | arm_kprobe_decode_insn(kprobe_opcode_t insn, struct arch_specific_insn *asi) | 1627 | arm_kprobe_decode_insn(kprobe_opcode_t insn, struct arch_specific_insn *asi) |
| 1403 | { | 1628 | { |
| 1629 | asi->insn_check_cc = condition_checks[insn>>28]; | ||
| 1404 | asi->insn[1] = KPROBE_RETURN_INSTRUCTION; | 1630 | asi->insn[1] = KPROBE_RETURN_INSTRUCTION; |
| 1405 | 1631 | ||
| 1406 | if ((insn & 0xf0000000) == 0xf0000000) { | 1632 | if ((insn & 0xf0000000) == 0xf0000000) |
| 1407 | 1633 | ||
| 1408 | return space_1111(insn, asi); | 1634 | return space_1111(insn, asi); |
| 1409 | 1635 | ||
| 1410 | } else if ((insn & 0x0e000000) == 0x00000000) { | 1636 | else if ((insn & 0x0e000000) == 0x00000000) |
| 1411 | 1637 | ||
| 1412 | return space_cccc_000x(insn, asi); | 1638 | return space_cccc_000x(insn, asi); |
| 1413 | 1639 | ||
| 1414 | } else if ((insn & 0x0e000000) == 0x02000000) { | 1640 | else if ((insn & 0x0e000000) == 0x02000000) |
| 1415 | 1641 | ||
| 1416 | return space_cccc_001x(insn, asi); | 1642 | return space_cccc_001x(insn, asi); |
| 1417 | 1643 | ||
| 1418 | } else if ((insn & 0x0f000010) == 0x06000010) { | 1644 | else if ((insn & 0x0f000010) == 0x06000010) |
| 1419 | 1645 | ||
| 1420 | return space_cccc_0110__1(insn, asi); | 1646 | return space_cccc_0110__1(insn, asi); |
| 1421 | 1647 | ||
| 1422 | } else if ((insn & 0x0f000010) == 0x07000010) { | 1648 | else if ((insn & 0x0f000010) == 0x07000010) |
| 1423 | 1649 | ||
| 1424 | return space_cccc_0111__1(insn, asi); | 1650 | return space_cccc_0111__1(insn, asi); |
| 1425 | 1651 | ||
| 1426 | } else if ((insn & 0x0c000000) == 0x04000000) { | 1652 | else if ((insn & 0x0c000000) == 0x04000000) |
| 1427 | 1653 | ||
| 1428 | return space_cccc_01xx(insn, asi); | 1654 | return space_cccc_01xx(insn, asi); |
| 1429 | 1655 | ||
| 1430 | } else if ((insn & 0x0e000000) == 0x08000000) { | 1656 | else if ((insn & 0x0e000000) == 0x08000000) |
| 1431 | 1657 | ||
| 1432 | return space_cccc_100x(insn, asi); | 1658 | return space_cccc_100x(insn, asi); |
| 1433 | 1659 | ||
| 1434 | } else if ((insn & 0x0e000000) == 0x0a000000) { | 1660 | else if ((insn & 0x0e000000) == 0x0a000000) |
| 1435 | 1661 | ||
| 1436 | return space_cccc_101x(insn, asi); | 1662 | return space_cccc_101x(insn, asi); |
| 1437 | 1663 | ||
| 1438 | } else if ((insn & 0x0fe00000) == 0x0c400000) { | 1664 | return space_cccc_11xx(insn, asi); |
| 1439 | |||
| 1440 | return space_cccc_1100_010x(insn, asi); | ||
| 1441 | |||
| 1442 | } else if ((insn & 0x0e000000) == 0x0c000000) { | ||
| 1443 | |||
| 1444 | return space_cccc_110x(insn, asi); | ||
| 1445 | |||
| 1446 | } | ||
| 1447 | |||
| 1448 | return space_cccc_111x(insn, asi); | ||
| 1449 | } | 1665 | } |
| 1450 | 1666 | ||
| 1451 | void __init arm_kprobe_decode_init(void) | 1667 | void __init arm_kprobe_decode_init(void) |
| 1452 | { | 1668 | { |
| 1453 | find_str_pc_offset(); | 1669 | find_str_pc_offset(); |
| 1454 | } | 1670 | } |
| 1455 | |||
| 1456 | |||
| 1457 | /* | ||
| 1458 | * All ARM instructions listed below. | ||
| 1459 | * | ||
| 1460 | * Instructions and their general purpose registers are given. | ||
| 1461 | * If a particular register may not use R15, it is prefixed with a "!". | ||
| 1462 | * If marked with a "*" means the value returned by reading R15 | ||
| 1463 | * is implementation defined. | ||
| 1464 | * | ||
| 1465 | * ADC/ADD/AND/BIC/CMN/CMP/EOR/MOV/MVN/ORR/RSB/RSC/SBC/SUB/TEQ | ||
| 1466 | * TST: Rd, Rn, Rm, !Rs | ||
| 1467 | * BX: Rm | ||
| 1468 | * BLX(2): !Rm | ||
| 1469 | * BX: Rm (R15 legal, but discouraged) | ||
| 1470 | * BXJ: !Rm, | ||
| 1471 | * CLZ: !Rd, !Rm | ||
| 1472 | * CPY: Rd, Rm | ||
| 1473 | * LDC/2,STC/2 immediate offset & unindex: Rn | ||
| 1474 | * LDC/2,STC/2 immediate pre/post-indexed: !Rn | ||
| 1475 | * LDM(1/3): !Rn, register_list | ||
| 1476 | * LDM(2): !Rn, !register_list | ||
| 1477 | * LDR,STR,PLD immediate offset: Rd, Rn | ||
| 1478 | * LDR,STR,PLD register offset: Rd, Rn, !Rm | ||
| 1479 | * LDR,STR,PLD scaled register offset: Rd, !Rn, !Rm | ||
| 1480 | * LDR,STR immediate pre/post-indexed: Rd, !Rn | ||
| 1481 | * LDR,STR register pre/post-indexed: Rd, !Rn, !Rm | ||
| 1482 | * LDR,STR scaled register pre/post-indexed: Rd, !Rn, !Rm | ||
| 1483 | * LDRB,STRB immediate offset: !Rd, Rn | ||
| 1484 | * LDRB,STRB register offset: !Rd, Rn, !Rm | ||
| 1485 | * LDRB,STRB scaled register offset: !Rd, !Rn, !Rm | ||
| 1486 | * LDRB,STRB immediate pre/post-indexed: !Rd, !Rn | ||
| 1487 | * LDRB,STRB register pre/post-indexed: !Rd, !Rn, !Rm | ||
| 1488 | * LDRB,STRB scaled register pre/post-indexed: !Rd, !Rn, !Rm | ||
| 1489 | * LDRT,LDRBT,STRBT immediate pre/post-indexed: !Rd, !Rn | ||
| 1490 | * LDRT,LDRBT,STRBT register pre/post-indexed: !Rd, !Rn, !Rm | ||
| 1491 | * LDRT,LDRBT,STRBT scaled register pre/post-indexed: !Rd, !Rn, !Rm | ||
| 1492 | * LDRH/SH/SB/D,STRH/SH/SB/D immediate offset: !Rd, Rn | ||
| 1493 | * LDRH/SH/SB/D,STRH/SH/SB/D register offset: !Rd, Rn, !Rm | ||
| 1494 | * LDRH/SH/SB/D,STRH/SH/SB/D immediate pre/post-indexed: !Rd, !Rn | ||
| 1495 | * LDRH/SH/SB/D,STRH/SH/SB/D register pre/post-indexed: !Rd, !Rn, !Rm | ||
| 1496 | * LDREX: !Rd, !Rn | ||
| 1497 | * MCR/2: !Rd | ||
| 1498 | * MCRR/2,MRRC/2: !Rd, !Rn | ||
| 1499 | * MLA: !Rd, !Rn, !Rm, !Rs | ||
| 1500 | * MOV: Rd | ||
| 1501 | * MRC/2: !Rd (if Rd==15, only changes cond codes, not the register) | ||
| 1502 | * MRS,MSR: !Rd | ||
| 1503 | * MUL: !Rd, !Rm, !Rs | ||
| 1504 | * PKH{BT,TB}: !Rd, !Rn, !Rm | ||
| 1505 | * QDADD,[U]QADD/16/8/SUBX: !Rd, !Rm, !Rn | ||
| 1506 | * QDSUB,[U]QSUB/16/8/ADDX: !Rd, !Rm, !Rn | ||
| 1507 | * REV/16/SH: !Rd, !Rm | ||
| 1508 | * RFE: !Rn | ||
| 1509 | * {S,U}[H]ADD{16,8,SUBX},{S,U}[H]SUB{16,8,ADDX}: !Rd, !Rn, !Rm | ||
| 1510 | * SEL: !Rd, !Rn, !Rm | ||
| 1511 | * SMLA<x><y>,SMLA{D,W<y>},SMLSD,SMML{A,S}: !Rd, !Rn, !Rm, !Rs | ||
| 1512 | * SMLAL<x><y>,SMLA{D,LD},SMLSLD,SMMULL,SMULW<y>: !RdHi, !RdLo, !Rm, !Rs | ||
| 1513 | * SMMUL,SMUAD,SMUL<x><y>,SMUSD: !Rd, !Rm, !Rs | ||
| 1514 | * SSAT/16: !Rd, !Rm | ||
| 1515 | * STM(1/2): !Rn, register_list* (R15 in reg list not recommended) | ||
| 1516 | * STRT immediate pre/post-indexed: Rd*, !Rn | ||
| 1517 | * STRT register pre/post-indexed: Rd*, !Rn, !Rm | ||
| 1518 | * STRT scaled register pre/post-indexed: Rd*, !Rn, !Rm | ||
| 1519 | * STREX: !Rd, !Rn, !Rm | ||
| 1520 | * SWP/B: !Rd, !Rn, !Rm | ||
| 1521 | * {S,U}XTA{B,B16,H}: !Rd, !Rn, !Rm | ||
| 1522 | * {S,U}XT{B,B16,H}: !Rd, !Rm | ||
| 1523 | * UM{AA,LA,UL}L: !RdHi, !RdLo, !Rm, !Rs | ||
| 1524 | * USA{D8,A8,T,T16}: !Rd, !Rm, !Rs | ||
| 1525 | * | ||
| 1526 | * May transfer control by writing R15 (possible mode changes or alternate | ||
| 1527 | * mode accesses marked by "*"): | ||
| 1528 | * ALU op (* with s-bit), B, BL, BKPT, BLX(1/2), BX, BXJ, CPS*, CPY, | ||
| 1529 | * LDM(1), LDM(2/3)*, LDR, MOV, RFE*, SWI* | ||
| 1530 | * | ||
| 1531 | * Instructions that do not take general registers, nor transfer control: | ||
| 1532 | * CDP/2, SETEND, SRS* | ||
| 1533 | */ | ||
diff --git a/arch/arm/kernel/kprobes.c b/arch/arm/kernel/kprobes.c index 2ba7deb3072e..1656c87501c0 100644 --- a/arch/arm/kernel/kprobes.c +++ b/arch/arm/kernel/kprobes.c | |||
| @@ -134,7 +134,8 @@ static void __kprobes singlestep(struct kprobe *p, struct pt_regs *regs, | |||
| 134 | struct kprobe_ctlblk *kcb) | 134 | struct kprobe_ctlblk *kcb) |
| 135 | { | 135 | { |
| 136 | regs->ARM_pc += 4; | 136 | regs->ARM_pc += 4; |
| 137 | p->ainsn.insn_handler(p, regs); | 137 | if (p->ainsn.insn_check_cc(regs->ARM_cpsr)) |
| 138 | p->ainsn.insn_handler(p, regs); | ||
| 138 | } | 139 | } |
| 139 | 140 | ||
| 140 | /* | 141 | /* |
diff --git a/arch/arm/kernel/perf_event.c b/arch/arm/kernel/perf_event.c index 979da3947f42..139e3c827369 100644 --- a/arch/arm/kernel/perf_event.c +++ b/arch/arm/kernel/perf_event.c | |||
| @@ -746,7 +746,8 @@ perf_callchain_user(struct perf_callchain_entry *entry, struct pt_regs *regs) | |||
| 746 | 746 | ||
| 747 | tail = (struct frame_tail __user *)regs->ARM_fp - 1; | 747 | tail = (struct frame_tail __user *)regs->ARM_fp - 1; |
| 748 | 748 | ||
| 749 | while (tail && !((unsigned long)tail & 0x3)) | 749 | while ((entry->nr < PERF_MAX_STACK_DEPTH) && |
| 750 | tail && !((unsigned long)tail & 0x3)) | ||
| 750 | tail = user_backtrace(tail, entry); | 751 | tail = user_backtrace(tail, entry); |
| 751 | } | 752 | } |
| 752 | 753 | ||
diff --git a/arch/arm/kernel/smp.c b/arch/arm/kernel/smp.c index 8fe05ad932e4..f29b8a29b174 100644 --- a/arch/arm/kernel/smp.c +++ b/arch/arm/kernel/smp.c | |||
| @@ -479,7 +479,7 @@ static void broadcast_timer_set_mode(enum clock_event_mode mode, | |||
| 479 | { | 479 | { |
| 480 | } | 480 | } |
| 481 | 481 | ||
| 482 | static void broadcast_timer_setup(struct clock_event_device *evt) | 482 | static void __cpuinit broadcast_timer_setup(struct clock_event_device *evt) |
| 483 | { | 483 | { |
| 484 | evt->name = "dummy_timer"; | 484 | evt->name = "dummy_timer"; |
| 485 | evt->features = CLOCK_EVT_FEAT_ONESHOT | | 485 | evt->features = CLOCK_EVT_FEAT_ONESHOT | |
diff --git a/arch/arm/kernel/sys_oabi-compat.c b/arch/arm/kernel/sys_oabi-compat.c index 4ad8da15ef2b..af0aaebf4de6 100644 --- a/arch/arm/kernel/sys_oabi-compat.c +++ b/arch/arm/kernel/sys_oabi-compat.c | |||
| @@ -311,7 +311,7 @@ asmlinkage long sys_oabi_semtimedop(int semid, | |||
| 311 | long err; | 311 | long err; |
| 312 | int i; | 312 | int i; |
| 313 | 313 | ||
| 314 | if (nsops < 1) | 314 | if (nsops < 1 || nsops > SEMOPM) |
| 315 | return -EINVAL; | 315 | return -EINVAL; |
| 316 | sops = kmalloc(sizeof(*sops) * nsops, GFP_KERNEL); | 316 | sops = kmalloc(sizeof(*sops) * nsops, GFP_KERNEL); |
| 317 | if (!sops) | 317 | if (!sops) |
diff --git a/arch/arm/mach-davinci/Kconfig b/arch/arm/mach-davinci/Kconfig index 32f147998cd9..c0deacae778d 100644 --- a/arch/arm/mach-davinci/Kconfig +++ b/arch/arm/mach-davinci/Kconfig | |||
| @@ -63,6 +63,7 @@ config MACH_DAVINCI_EVM | |||
| 63 | depends on ARCH_DAVINCI_DM644x | 63 | depends on ARCH_DAVINCI_DM644x |
| 64 | select MISC_DEVICES | 64 | select MISC_DEVICES |
| 65 | select EEPROM_AT24 | 65 | select EEPROM_AT24 |
| 66 | select I2C | ||
| 66 | help | 67 | help |
| 67 | Configure this option to specify the whether the board used | 68 | Configure this option to specify the whether the board used |
| 68 | for development is a DM644x EVM | 69 | for development is a DM644x EVM |
| @@ -72,6 +73,7 @@ config MACH_SFFSDR | |||
| 72 | depends on ARCH_DAVINCI_DM644x | 73 | depends on ARCH_DAVINCI_DM644x |
| 73 | select MISC_DEVICES | 74 | select MISC_DEVICES |
| 74 | select EEPROM_AT24 | 75 | select EEPROM_AT24 |
| 76 | select I2C | ||
| 75 | help | 77 | help |
| 76 | Say Y here to select the Lyrtech Small Form Factor | 78 | Say Y here to select the Lyrtech Small Form Factor |
| 77 | Software Defined Radio (SFFSDR) board. | 79 | Software Defined Radio (SFFSDR) board. |
| @@ -105,6 +107,7 @@ config MACH_DAVINCI_DM6467_EVM | |||
| 105 | select MACH_DAVINCI_DM6467TEVM | 107 | select MACH_DAVINCI_DM6467TEVM |
| 106 | select MISC_DEVICES | 108 | select MISC_DEVICES |
| 107 | select EEPROM_AT24 | 109 | select EEPROM_AT24 |
| 110 | select I2C | ||
| 108 | help | 111 | help |
| 109 | Configure this option to specify the whether the board used | 112 | Configure this option to specify the whether the board used |
| 110 | for development is a DM6467 EVM | 113 | for development is a DM6467 EVM |
| @@ -118,6 +121,7 @@ config MACH_DAVINCI_DM365_EVM | |||
| 118 | depends on ARCH_DAVINCI_DM365 | 121 | depends on ARCH_DAVINCI_DM365 |
| 119 | select MISC_DEVICES | 122 | select MISC_DEVICES |
| 120 | select EEPROM_AT24 | 123 | select EEPROM_AT24 |
| 124 | select I2C | ||
| 121 | help | 125 | help |
| 122 | Configure this option to specify whether the board used | 126 | Configure this option to specify whether the board used |
| 123 | for development is a DM365 EVM | 127 | for development is a DM365 EVM |
| @@ -129,6 +133,7 @@ config MACH_DAVINCI_DA830_EVM | |||
| 129 | select GPIO_PCF857X | 133 | select GPIO_PCF857X |
| 130 | select MISC_DEVICES | 134 | select MISC_DEVICES |
| 131 | select EEPROM_AT24 | 135 | select EEPROM_AT24 |
| 136 | select I2C | ||
| 132 | help | 137 | help |
| 133 | Say Y here to select the TI DA830/OMAP-L137/AM17x Evaluation Module. | 138 | Say Y here to select the TI DA830/OMAP-L137/AM17x Evaluation Module. |
| 134 | 139 | ||
| @@ -205,6 +210,7 @@ config MACH_MITYOMAPL138 | |||
| 205 | depends on ARCH_DAVINCI_DA850 | 210 | depends on ARCH_DAVINCI_DA850 |
| 206 | select MISC_DEVICES | 211 | select MISC_DEVICES |
| 207 | select EEPROM_AT24 | 212 | select EEPROM_AT24 |
| 213 | select I2C | ||
| 208 | help | 214 | help |
| 209 | Say Y here to select the Critical Link MityDSP-L138/MityARM-1808 | 215 | Say Y here to select the Critical Link MityDSP-L138/MityARM-1808 |
| 210 | System on Module. Information on this SoM may be found at | 216 | System on Module. Information on this SoM may be found at |
diff --git a/arch/arm/mach-davinci/board-mityomapl138.c b/arch/arm/mach-davinci/board-mityomapl138.c index 2aa79c54f98e..606a6f27ed6c 100644 --- a/arch/arm/mach-davinci/board-mityomapl138.c +++ b/arch/arm/mach-davinci/board-mityomapl138.c | |||
| @@ -29,7 +29,7 @@ | |||
| 29 | #include <mach/mux.h> | 29 | #include <mach/mux.h> |
| 30 | #include <mach/spi.h> | 30 | #include <mach/spi.h> |
| 31 | 31 | ||
| 32 | #define MITYOMAPL138_PHY_ID "0:03" | 32 | #define MITYOMAPL138_PHY_ID "" |
| 33 | 33 | ||
| 34 | #define FACTORY_CONFIG_MAGIC 0x012C0138 | 34 | #define FACTORY_CONFIG_MAGIC 0x012C0138 |
| 35 | #define FACTORY_CONFIG_VERSION 0x00010001 | 35 | #define FACTORY_CONFIG_VERSION 0x00010001 |
| @@ -414,7 +414,7 @@ static struct resource mityomapl138_nandflash_resource[] = { | |||
| 414 | 414 | ||
| 415 | static struct platform_device mityomapl138_nandflash_device = { | 415 | static struct platform_device mityomapl138_nandflash_device = { |
| 416 | .name = "davinci_nand", | 416 | .name = "davinci_nand", |
| 417 | .id = 0, | 417 | .id = 1, |
| 418 | .dev = { | 418 | .dev = { |
| 419 | .platform_data = &mityomapl138_nandflash_data, | 419 | .platform_data = &mityomapl138_nandflash_data, |
| 420 | }, | 420 | }, |
diff --git a/arch/arm/mach-davinci/devices-da8xx.c b/arch/arm/mach-davinci/devices-da8xx.c index 625d4b66718b..58a02dc7b15a 100644 --- a/arch/arm/mach-davinci/devices-da8xx.c +++ b/arch/arm/mach-davinci/devices-da8xx.c | |||
| @@ -39,7 +39,8 @@ | |||
| 39 | #define DA8XX_GPIO_BASE 0x01e26000 | 39 | #define DA8XX_GPIO_BASE 0x01e26000 |
| 40 | #define DA8XX_I2C1_BASE 0x01e28000 | 40 | #define DA8XX_I2C1_BASE 0x01e28000 |
| 41 | #define DA8XX_SPI0_BASE 0x01c41000 | 41 | #define DA8XX_SPI0_BASE 0x01c41000 |
| 42 | #define DA8XX_SPI1_BASE 0x01f0e000 | 42 | #define DA830_SPI1_BASE 0x01e12000 |
| 43 | #define DA850_SPI1_BASE 0x01f0e000 | ||
| 43 | 44 | ||
| 44 | #define DA8XX_EMAC_CTRL_REG_OFFSET 0x3000 | 45 | #define DA8XX_EMAC_CTRL_REG_OFFSET 0x3000 |
| 45 | #define DA8XX_EMAC_MOD_REG_OFFSET 0x2000 | 46 | #define DA8XX_EMAC_MOD_REG_OFFSET 0x2000 |
| @@ -762,8 +763,8 @@ static struct resource da8xx_spi0_resources[] = { | |||
| 762 | 763 | ||
| 763 | static struct resource da8xx_spi1_resources[] = { | 764 | static struct resource da8xx_spi1_resources[] = { |
| 764 | [0] = { | 765 | [0] = { |
| 765 | .start = DA8XX_SPI1_BASE, | 766 | .start = DA830_SPI1_BASE, |
| 766 | .end = DA8XX_SPI1_BASE + SZ_4K - 1, | 767 | .end = DA830_SPI1_BASE + SZ_4K - 1, |
| 767 | .flags = IORESOURCE_MEM, | 768 | .flags = IORESOURCE_MEM, |
| 768 | }, | 769 | }, |
| 769 | [1] = { | 770 | [1] = { |
| @@ -832,5 +833,10 @@ int __init da8xx_register_spi(int instance, struct spi_board_info *info, | |||
| 832 | 833 | ||
| 833 | da8xx_spi_pdata[instance].num_chipselect = len; | 834 | da8xx_spi_pdata[instance].num_chipselect = len; |
| 834 | 835 | ||
| 836 | if (instance == 1 && cpu_is_davinci_da850()) { | ||
| 837 | da8xx_spi1_resources[0].start = DA850_SPI1_BASE; | ||
| 838 | da8xx_spi1_resources[0].end = DA850_SPI1_BASE + SZ_4K - 1; | ||
| 839 | } | ||
| 840 | |||
| 835 | return platform_device_register(&da8xx_spi_device[instance]); | 841 | return platform_device_register(&da8xx_spi_device[instance]); |
| 836 | } | 842 | } |
diff --git a/arch/arm/mach-davinci/dm355.c b/arch/arm/mach-davinci/dm355.c index f68012239641..a3a94e9c9378 100644 --- a/arch/arm/mach-davinci/dm355.c +++ b/arch/arm/mach-davinci/dm355.c | |||
| @@ -314,7 +314,7 @@ static struct clk timer2_clk = { | |||
| 314 | .name = "timer2", | 314 | .name = "timer2", |
| 315 | .parent = &pll1_aux_clk, | 315 | .parent = &pll1_aux_clk, |
| 316 | .lpsc = DAVINCI_LPSC_TIMER2, | 316 | .lpsc = DAVINCI_LPSC_TIMER2, |
| 317 | .usecount = 1, /* REVISIT: why can't' this be disabled? */ | 317 | .usecount = 1, /* REVISIT: why can't this be disabled? */ |
| 318 | }; | 318 | }; |
| 319 | 319 | ||
| 320 | static struct clk timer3_clk = { | 320 | static struct clk timer3_clk = { |
diff --git a/arch/arm/mach-davinci/dm644x.c b/arch/arm/mach-davinci/dm644x.c index 5f8a65424184..4c82c2716293 100644 --- a/arch/arm/mach-davinci/dm644x.c +++ b/arch/arm/mach-davinci/dm644x.c | |||
| @@ -274,7 +274,7 @@ static struct clk timer2_clk = { | |||
| 274 | .name = "timer2", | 274 | .name = "timer2", |
| 275 | .parent = &pll1_aux_clk, | 275 | .parent = &pll1_aux_clk, |
| 276 | .lpsc = DAVINCI_LPSC_TIMER2, | 276 | .lpsc = DAVINCI_LPSC_TIMER2, |
| 277 | .usecount = 1, /* REVISIT: why can't' this be disabled? */ | 277 | .usecount = 1, /* REVISIT: why can't this be disabled? */ |
| 278 | }; | 278 | }; |
| 279 | 279 | ||
| 280 | static struct clk_lookup dm644x_clks[] = { | 280 | static struct clk_lookup dm644x_clks[] = { |
diff --git a/arch/arm/mach-davinci/include/mach/debug-macro.S b/arch/arm/mach-davinci/include/mach/debug-macro.S index 9f1befc5ac38..f8b7ea4f6235 100644 --- a/arch/arm/mach-davinci/include/mach/debug-macro.S +++ b/arch/arm/mach-davinci/include/mach/debug-macro.S | |||
| @@ -24,6 +24,9 @@ | |||
| 24 | 24 | ||
| 25 | #define UART_SHIFT 2 | 25 | #define UART_SHIFT 2 |
| 26 | 26 | ||
| 27 | #define davinci_uart_v2p(x) ((x) - PAGE_OFFSET + PLAT_PHYS_OFFSET) | ||
| 28 | #define davinci_uart_p2v(x) ((x) - PLAT_PHYS_OFFSET + PAGE_OFFSET) | ||
| 29 | |||
| 27 | .pushsection .data | 30 | .pushsection .data |
| 28 | davinci_uart_phys: .word 0 | 31 | davinci_uart_phys: .word 0 |
| 29 | davinci_uart_virt: .word 0 | 32 | davinci_uart_virt: .word 0 |
| @@ -34,7 +37,7 @@ davinci_uart_virt: .word 0 | |||
| 34 | /* Use davinci_uart_phys/virt if already configured */ | 37 | /* Use davinci_uart_phys/virt if already configured */ |
| 35 | 10: mrc p15, 0, \rp, c1, c0 | 38 | 10: mrc p15, 0, \rp, c1, c0 |
| 36 | tst \rp, #1 @ MMU enabled? | 39 | tst \rp, #1 @ MMU enabled? |
| 37 | ldreq \rp, =__virt_to_phys(davinci_uart_phys) | 40 | ldreq \rp, =davinci_uart_v2p(davinci_uart_phys) |
| 38 | ldrne \rp, =davinci_uart_phys | 41 | ldrne \rp, =davinci_uart_phys |
| 39 | add \rv, \rp, #4 @ davinci_uart_virt | 42 | add \rv, \rp, #4 @ davinci_uart_virt |
| 40 | ldr \rp, [\rp, #0] | 43 | ldr \rp, [\rp, #0] |
| @@ -48,18 +51,18 @@ davinci_uart_virt: .word 0 | |||
| 48 | tst \rp, #1 @ MMU enabled? | 51 | tst \rp, #1 @ MMU enabled? |
| 49 | 52 | ||
| 50 | /* Copy uart phys address from decompressor uart info */ | 53 | /* Copy uart phys address from decompressor uart info */ |
| 51 | ldreq \rv, =__virt_to_phys(davinci_uart_phys) | 54 | ldreq \rv, =davinci_uart_v2p(davinci_uart_phys) |
| 52 | ldrne \rv, =davinci_uart_phys | 55 | ldrne \rv, =davinci_uart_phys |
| 53 | ldreq \rp, =DAVINCI_UART_INFO | 56 | ldreq \rp, =DAVINCI_UART_INFO |
| 54 | ldrne \rp, =__phys_to_virt(DAVINCI_UART_INFO) | 57 | ldrne \rp, =davinci_uart_p2v(DAVINCI_UART_INFO) |
| 55 | ldr \rp, [\rp, #0] | 58 | ldr \rp, [\rp, #0] |
| 56 | str \rp, [\rv] | 59 | str \rp, [\rv] |
| 57 | 60 | ||
| 58 | /* Copy uart virt address from decompressor uart info */ | 61 | /* Copy uart virt address from decompressor uart info */ |
| 59 | ldreq \rv, =__virt_to_phys(davinci_uart_virt) | 62 | ldreq \rv, =davinci_uart_v2p(davinci_uart_virt) |
| 60 | ldrne \rv, =davinci_uart_virt | 63 | ldrne \rv, =davinci_uart_virt |
| 61 | ldreq \rp, =DAVINCI_UART_INFO | 64 | ldreq \rp, =DAVINCI_UART_INFO |
| 62 | ldrne \rp, =__phys_to_virt(DAVINCI_UART_INFO) | 65 | ldrne \rp, =davinci_uart_p2v(DAVINCI_UART_INFO) |
| 63 | ldr \rp, [\rp, #4] | 66 | ldr \rp, [\rp, #4] |
| 64 | str \rp, [\rv] | 67 | str \rp, [\rv] |
| 65 | 68 | ||
diff --git a/arch/arm/mach-davinci/include/mach/serial.h b/arch/arm/mach-davinci/include/mach/serial.h index 8051110b8ac3..c9e6ce185a66 100644 --- a/arch/arm/mach-davinci/include/mach/serial.h +++ b/arch/arm/mach-davinci/include/mach/serial.h | |||
| @@ -22,7 +22,7 @@ | |||
| 22 | * | 22 | * |
| 23 | * This area sits just below the page tables (see arch/arm/kernel/head.S). | 23 | * This area sits just below the page tables (see arch/arm/kernel/head.S). |
| 24 | */ | 24 | */ |
| 25 | #define DAVINCI_UART_INFO (PHYS_OFFSET + 0x3ff8) | 25 | #define DAVINCI_UART_INFO (PLAT_PHYS_OFFSET + 0x3ff8) |
| 26 | 26 | ||
| 27 | #define DAVINCI_UART0_BASE (IO_PHYS + 0x20000) | 27 | #define DAVINCI_UART0_BASE (IO_PHYS + 0x20000) |
| 28 | #define DAVINCI_UART1_BASE (IO_PHYS + 0x20400) | 28 | #define DAVINCI_UART1_BASE (IO_PHYS + 0x20400) |
diff --git a/arch/arm/mach-mx3/mach-vpr200.c b/arch/arm/mach-mx3/mach-vpr200.c index 2cf390fbd980..47a69cbc31a8 100644 --- a/arch/arm/mach-mx3/mach-vpr200.c +++ b/arch/arm/mach-mx3/mach-vpr200.c | |||
| @@ -257,11 +257,16 @@ static const struct fsl_usb2_platform_data otg_device_pdata __initconst = { | |||
| 257 | .workaround = FLS_USB2_WORKAROUND_ENGCM09152, | 257 | .workaround = FLS_USB2_WORKAROUND_ENGCM09152, |
| 258 | }; | 258 | }; |
| 259 | 259 | ||
| 260 | static int vpr200_usbh_init(struct platform_device *pdev) | ||
| 261 | { | ||
| 262 | return mx35_initialize_usb_hw(pdev->id, | ||
| 263 | MXC_EHCI_INTERFACE_SINGLE_UNI | MXC_EHCI_INTERNAL_PHY); | ||
| 264 | } | ||
| 265 | |||
| 260 | /* USB HOST config */ | 266 | /* USB HOST config */ |
| 261 | static const struct mxc_usbh_platform_data usb_host_pdata __initconst = { | 267 | static const struct mxc_usbh_platform_data usb_host_pdata __initconst = { |
| 262 | .portsc = MXC_EHCI_MODE_SERIAL, | 268 | .init = vpr200_usbh_init, |
| 263 | .flags = MXC_EHCI_INTERFACE_SINGLE_UNI | | 269 | .portsc = MXC_EHCI_MODE_SERIAL, |
| 264 | MXC_EHCI_INTERNAL_PHY, | ||
| 265 | }; | 270 | }; |
| 266 | 271 | ||
| 267 | static struct platform_device *devices[] __initdata = { | 272 | static struct platform_device *devices[] __initdata = { |
diff --git a/arch/arm/mach-mx5/board-mx53_loco.c b/arch/arm/mach-mx5/board-mx53_loco.c index 10a1bea10548..6206b1191fe8 100644 --- a/arch/arm/mach-mx5/board-mx53_loco.c +++ b/arch/arm/mach-mx5/board-mx53_loco.c | |||
| @@ -193,7 +193,7 @@ static iomux_v3_cfg_t mx53_loco_pads[] = { | |||
| 193 | .wakeup = wake, \ | 193 | .wakeup = wake, \ |
| 194 | } | 194 | } |
| 195 | 195 | ||
| 196 | static const struct gpio_keys_button loco_buttons[] __initconst = { | 196 | static struct gpio_keys_button loco_buttons[] = { |
| 197 | GPIO_BUTTON(MX53_LOCO_POWER, KEY_POWER, 1, "power", 0), | 197 | GPIO_BUTTON(MX53_LOCO_POWER, KEY_POWER, 1, "power", 0), |
| 198 | GPIO_BUTTON(MX53_LOCO_UI1, KEY_VOLUMEUP, 1, "volume-up", 0), | 198 | GPIO_BUTTON(MX53_LOCO_UI1, KEY_VOLUMEUP, 1, "volume-up", 0), |
| 199 | GPIO_BUTTON(MX53_LOCO_UI2, KEY_VOLUMEDOWN, 1, "volume-down", 0), | 199 | GPIO_BUTTON(MX53_LOCO_UI2, KEY_VOLUMEDOWN, 1, "volume-down", 0), |
diff --git a/arch/arm/mach-mxs/clock-mx28.c b/arch/arm/mach-mxs/clock-mx28.c index 1ad97fed1e94..5dcc59d5b9ec 100644 --- a/arch/arm/mach-mxs/clock-mx28.c +++ b/arch/arm/mach-mxs/clock-mx28.c | |||
| @@ -295,11 +295,11 @@ static int name##_set_rate(struct clk *clk, unsigned long rate) \ | |||
| 295 | unsigned long diff, parent_rate, calc_rate; \ | 295 | unsigned long diff, parent_rate, calc_rate; \ |
| 296 | int i; \ | 296 | int i; \ |
| 297 | \ | 297 | \ |
| 298 | parent_rate = clk_get_rate(clk->parent); \ | ||
| 299 | div_max = BM_CLKCTRL_##dr##_DIV >> BP_CLKCTRL_##dr##_DIV; \ | 298 | div_max = BM_CLKCTRL_##dr##_DIV >> BP_CLKCTRL_##dr##_DIV; \ |
| 300 | bm_busy = BM_CLKCTRL_##dr##_BUSY; \ | 299 | bm_busy = BM_CLKCTRL_##dr##_BUSY; \ |
| 301 | \ | 300 | \ |
| 302 | if (clk->parent == &ref_xtal_clk) { \ | 301 | if (clk->parent == &ref_xtal_clk) { \ |
| 302 | parent_rate = clk_get_rate(clk->parent); \ | ||
| 303 | div = DIV_ROUND_UP(parent_rate, rate); \ | 303 | div = DIV_ROUND_UP(parent_rate, rate); \ |
| 304 | if (clk == &cpu_clk) { \ | 304 | if (clk == &cpu_clk) { \ |
| 305 | div_max = BM_CLKCTRL_CPU_DIV_XTAL >> \ | 305 | div_max = BM_CLKCTRL_CPU_DIV_XTAL >> \ |
| @@ -309,6 +309,11 @@ static int name##_set_rate(struct clk *clk, unsigned long rate) \ | |||
| 309 | if (div == 0 || div > div_max) \ | 309 | if (div == 0 || div > div_max) \ |
| 310 | return -EINVAL; \ | 310 | return -EINVAL; \ |
| 311 | } else { \ | 311 | } else { \ |
| 312 | /* \ | ||
| 313 | * hack alert: this block modifies clk->parent, too, \ | ||
| 314 | * so the base to use it the grand parent. \ | ||
| 315 | */ \ | ||
| 316 | parent_rate = clk_get_rate(clk->parent->parent); \ | ||
| 312 | rate >>= PARENT_RATE_SHIFT; \ | 317 | rate >>= PARENT_RATE_SHIFT; \ |
| 313 | parent_rate >>= PARENT_RATE_SHIFT; \ | 318 | parent_rate >>= PARENT_RATE_SHIFT; \ |
| 314 | diff = parent_rate; \ | 319 | diff = parent_rate; \ |
diff --git a/arch/arm/mach-omap2/Makefile b/arch/arm/mach-omap2/Makefile index a45cd6409686..512b15204450 100644 --- a/arch/arm/mach-omap2/Makefile +++ b/arch/arm/mach-omap2/Makefile | |||
| @@ -68,7 +68,7 @@ obj-$(CONFIG_OMAP_SMARTREFLEX) += sr_device.o smartreflex.o | |||
| 68 | obj-$(CONFIG_OMAP_SMARTREFLEX_CLASS3) += smartreflex-class3.o | 68 | obj-$(CONFIG_OMAP_SMARTREFLEX_CLASS3) += smartreflex-class3.o |
| 69 | 69 | ||
| 70 | AFLAGS_sleep24xx.o :=-Wa,-march=armv6 | 70 | AFLAGS_sleep24xx.o :=-Wa,-march=armv6 |
| 71 | AFLAGS_sleep34xx.o :=-Wa,-march=armv7-a | 71 | AFLAGS_sleep34xx.o :=-Wa,-march=armv7-a$(plus_sec) |
| 72 | 72 | ||
| 73 | ifeq ($(CONFIG_PM_VERBOSE),y) | 73 | ifeq ($(CONFIG_PM_VERBOSE),y) |
| 74 | CFLAGS_pm_bus.o += -DDEBUG | 74 | CFLAGS_pm_bus.o += -DDEBUG |
diff --git a/arch/arm/mach-omap2/board-rx51.c b/arch/arm/mach-omap2/board-rx51.c index e964895b80e8..f8ba20a14e62 100644 --- a/arch/arm/mach-omap2/board-rx51.c +++ b/arch/arm/mach-omap2/board-rx51.c | |||
| @@ -141,14 +141,19 @@ static void __init rx51_init(void) | |||
| 141 | static void __init rx51_map_io(void) | 141 | static void __init rx51_map_io(void) |
| 142 | { | 142 | { |
| 143 | omap2_set_globals_3xxx(); | 143 | omap2_set_globals_3xxx(); |
| 144 | rx51_video_mem_init(); | ||
| 145 | omap34xx_map_common_io(); | 144 | omap34xx_map_common_io(); |
| 146 | } | 145 | } |
| 147 | 146 | ||
| 147 | static void __init rx51_reserve(void) | ||
| 148 | { | ||
| 149 | rx51_video_mem_init(); | ||
| 150 | omap_reserve(); | ||
| 151 | } | ||
| 152 | |||
| 148 | MACHINE_START(NOKIA_RX51, "Nokia RX-51 board") | 153 | MACHINE_START(NOKIA_RX51, "Nokia RX-51 board") |
| 149 | /* Maintainer: Lauri Leukkunen <lauri.leukkunen@nokia.com> */ | 154 | /* Maintainer: Lauri Leukkunen <lauri.leukkunen@nokia.com> */ |
| 150 | .boot_params = 0x80000100, | 155 | .boot_params = 0x80000100, |
| 151 | .reserve = omap_reserve, | 156 | .reserve = rx51_reserve, |
| 152 | .map_io = rx51_map_io, | 157 | .map_io = rx51_map_io, |
| 153 | .init_early = rx51_init_early, | 158 | .init_early = rx51_init_early, |
| 154 | .init_irq = omap_init_irq, | 159 | .init_irq = omap_init_irq, |
diff --git a/arch/arm/mach-omap2/clock44xx_data.c b/arch/arm/mach-omap2/clock44xx_data.c index 276992d3b7fb..8c965671b4d4 100644 --- a/arch/arm/mach-omap2/clock44xx_data.c +++ b/arch/arm/mach-omap2/clock44xx_data.c | |||
| @@ -3116,14 +3116,9 @@ static struct omap_clk omap44xx_clks[] = { | |||
| 3116 | CLK(NULL, "dsp_fck", &dsp_fck, CK_443X), | 3116 | CLK(NULL, "dsp_fck", &dsp_fck, CK_443X), |
| 3117 | CLK("omapdss_dss", "sys_clk", &dss_sys_clk, CK_443X), | 3117 | CLK("omapdss_dss", "sys_clk", &dss_sys_clk, CK_443X), |
| 3118 | CLK("omapdss_dss", "tv_clk", &dss_tv_clk, CK_443X), | 3118 | CLK("omapdss_dss", "tv_clk", &dss_tv_clk, CK_443X), |
| 3119 | CLK("omapdss_dss", "dss_clk", &dss_dss_clk, CK_443X), | ||
| 3120 | CLK("omapdss_dss", "video_clk", &dss_48mhz_clk, CK_443X), | 3119 | CLK("omapdss_dss", "video_clk", &dss_48mhz_clk, CK_443X), |
| 3121 | CLK("omapdss_dss", "fck", &dss_fck, CK_443X), | 3120 | CLK("omapdss_dss", "fck", &dss_dss_clk, CK_443X), |
| 3122 | /* | 3121 | CLK("omapdss_dss", "ick", &dss_fck, CK_443X), |
| 3123 | * On OMAP4, DSS ick is a dummy clock; this is needed for compatibility | ||
| 3124 | * with OMAP2/3. | ||
| 3125 | */ | ||
| 3126 | CLK("omapdss_dss", "ick", &dummy_ck, CK_443X), | ||
| 3127 | CLK(NULL, "efuse_ctrl_cust_fck", &efuse_ctrl_cust_fck, CK_443X), | 3122 | CLK(NULL, "efuse_ctrl_cust_fck", &efuse_ctrl_cust_fck, CK_443X), |
| 3128 | CLK(NULL, "emif1_fck", &emif1_fck, CK_443X), | 3123 | CLK(NULL, "emif1_fck", &emif1_fck, CK_443X), |
| 3129 | CLK(NULL, "emif2_fck", &emif2_fck, CK_443X), | 3124 | CLK(NULL, "emif2_fck", &emif2_fck, CK_443X), |
diff --git a/arch/arm/mach-omap2/cm2xxx_3xxx.c b/arch/arm/mach-omap2/cm2xxx_3xxx.c index 9d0dec806e92..38830d8d4783 100644 --- a/arch/arm/mach-omap2/cm2xxx_3xxx.c +++ b/arch/arm/mach-omap2/cm2xxx_3xxx.c | |||
| @@ -247,6 +247,7 @@ struct omap3_cm_regs { | |||
| 247 | u32 per_cm_clksel; | 247 | u32 per_cm_clksel; |
| 248 | u32 emu_cm_clksel; | 248 | u32 emu_cm_clksel; |
| 249 | u32 emu_cm_clkstctrl; | 249 | u32 emu_cm_clkstctrl; |
| 250 | u32 pll_cm_autoidle; | ||
| 250 | u32 pll_cm_autoidle2; | 251 | u32 pll_cm_autoidle2; |
| 251 | u32 pll_cm_clksel4; | 252 | u32 pll_cm_clksel4; |
| 252 | u32 pll_cm_clksel5; | 253 | u32 pll_cm_clksel5; |
| @@ -319,6 +320,15 @@ void omap3_cm_save_context(void) | |||
| 319 | omap2_cm_read_mod_reg(OMAP3430_EMU_MOD, CM_CLKSEL1); | 320 | omap2_cm_read_mod_reg(OMAP3430_EMU_MOD, CM_CLKSEL1); |
| 320 | cm_context.emu_cm_clkstctrl = | 321 | cm_context.emu_cm_clkstctrl = |
| 321 | omap2_cm_read_mod_reg(OMAP3430_EMU_MOD, OMAP2_CM_CLKSTCTRL); | 322 | omap2_cm_read_mod_reg(OMAP3430_EMU_MOD, OMAP2_CM_CLKSTCTRL); |
| 323 | /* | ||
| 324 | * As per erratum i671, ROM code does not respect the PER DPLL | ||
| 325 | * programming scheme if CM_AUTOIDLE_PLL.AUTO_PERIPH_DPLL == 1. | ||
| 326 | * In this case, even though this register has been saved in | ||
| 327 | * scratchpad contents, we need to restore AUTO_PERIPH_DPLL | ||
| 328 | * by ourselves. So, we need to save it anyway. | ||
| 329 | */ | ||
| 330 | cm_context.pll_cm_autoidle = | ||
| 331 | omap2_cm_read_mod_reg(PLL_MOD, CM_AUTOIDLE); | ||
| 322 | cm_context.pll_cm_autoidle2 = | 332 | cm_context.pll_cm_autoidle2 = |
| 323 | omap2_cm_read_mod_reg(PLL_MOD, CM_AUTOIDLE2); | 333 | omap2_cm_read_mod_reg(PLL_MOD, CM_AUTOIDLE2); |
| 324 | cm_context.pll_cm_clksel4 = | 334 | cm_context.pll_cm_clksel4 = |
| @@ -441,6 +451,13 @@ void omap3_cm_restore_context(void) | |||
| 441 | CM_CLKSEL1); | 451 | CM_CLKSEL1); |
| 442 | omap2_cm_write_mod_reg(cm_context.emu_cm_clkstctrl, OMAP3430_EMU_MOD, | 452 | omap2_cm_write_mod_reg(cm_context.emu_cm_clkstctrl, OMAP3430_EMU_MOD, |
| 443 | OMAP2_CM_CLKSTCTRL); | 453 | OMAP2_CM_CLKSTCTRL); |
| 454 | /* | ||
| 455 | * As per erratum i671, ROM code does not respect the PER DPLL | ||
| 456 | * programming scheme if CM_AUTOIDLE_PLL.AUTO_PERIPH_DPLL == 1. | ||
| 457 | * In this case, we need to restore AUTO_PERIPH_DPLL by ourselves. | ||
| 458 | */ | ||
| 459 | omap2_cm_write_mod_reg(cm_context.pll_cm_autoidle, PLL_MOD, | ||
| 460 | CM_AUTOIDLE); | ||
| 444 | omap2_cm_write_mod_reg(cm_context.pll_cm_autoidle2, PLL_MOD, | 461 | omap2_cm_write_mod_reg(cm_context.pll_cm_autoidle2, PLL_MOD, |
| 445 | CM_AUTOIDLE2); | 462 | CM_AUTOIDLE2); |
| 446 | omap2_cm_write_mod_reg(cm_context.pll_cm_clksel4, PLL_MOD, | 463 | omap2_cm_write_mod_reg(cm_context.pll_cm_clksel4, PLL_MOD, |
diff --git a/arch/arm/mach-omap2/control.c b/arch/arm/mach-omap2/control.c index 695279419020..da53ba3917ca 100644 --- a/arch/arm/mach-omap2/control.c +++ b/arch/arm/mach-omap2/control.c | |||
| @@ -316,8 +316,14 @@ void omap3_save_scratchpad_contents(void) | |||
| 316 | omap2_cm_read_mod_reg(WKUP_MOD, CM_CLKSEL); | 316 | omap2_cm_read_mod_reg(WKUP_MOD, CM_CLKSEL); |
| 317 | prcm_block_contents.cm_clken_pll = | 317 | prcm_block_contents.cm_clken_pll = |
| 318 | omap2_cm_read_mod_reg(PLL_MOD, CM_CLKEN); | 318 | omap2_cm_read_mod_reg(PLL_MOD, CM_CLKEN); |
| 319 | /* | ||
| 320 | * As per erratum i671, ROM code does not respect the PER DPLL | ||
| 321 | * programming scheme if CM_AUTOIDLE_PLL..AUTO_PERIPH_DPLL == 1. | ||
| 322 | * Then, in anycase, clear these bits to avoid extra latencies. | ||
| 323 | */ | ||
| 319 | prcm_block_contents.cm_autoidle_pll = | 324 | prcm_block_contents.cm_autoidle_pll = |
| 320 | omap2_cm_read_mod_reg(PLL_MOD, OMAP3430_CM_AUTOIDLE_PLL); | 325 | omap2_cm_read_mod_reg(PLL_MOD, CM_AUTOIDLE) & |
| 326 | ~OMAP3430_AUTO_PERIPH_DPLL_MASK; | ||
| 321 | prcm_block_contents.cm_clksel1_pll = | 327 | prcm_block_contents.cm_clksel1_pll = |
| 322 | omap2_cm_read_mod_reg(PLL_MOD, OMAP3430_CM_CLKSEL1_PLL); | 328 | omap2_cm_read_mod_reg(PLL_MOD, OMAP3430_CM_CLKSEL1_PLL); |
| 323 | prcm_block_contents.cm_clksel2_pll = | 329 | prcm_block_contents.cm_clksel2_pll = |
diff --git a/arch/arm/mach-omap2/omap_hwmod_2420_data.c b/arch/arm/mach-omap2/omap_hwmod_2420_data.c index 8eb3ce1bbfbe..c4d0ae87d62a 100644 --- a/arch/arm/mach-omap2/omap_hwmod_2420_data.c +++ b/arch/arm/mach-omap2/omap_hwmod_2420_data.c | |||
| @@ -1639,6 +1639,7 @@ static struct omap_hwmod_ocp_if *omap2420_gpio1_slaves[] = { | |||
| 1639 | 1639 | ||
| 1640 | static struct omap_hwmod omap2420_gpio1_hwmod = { | 1640 | static struct omap_hwmod omap2420_gpio1_hwmod = { |
| 1641 | .name = "gpio1", | 1641 | .name = "gpio1", |
| 1642 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1642 | .mpu_irqs = omap242x_gpio1_irqs, | 1643 | .mpu_irqs = omap242x_gpio1_irqs, |
| 1643 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio1_irqs), | 1644 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio1_irqs), |
| 1644 | .main_clk = "gpios_fck", | 1645 | .main_clk = "gpios_fck", |
| @@ -1669,6 +1670,7 @@ static struct omap_hwmod_ocp_if *omap2420_gpio2_slaves[] = { | |||
| 1669 | 1670 | ||
| 1670 | static struct omap_hwmod omap2420_gpio2_hwmod = { | 1671 | static struct omap_hwmod omap2420_gpio2_hwmod = { |
| 1671 | .name = "gpio2", | 1672 | .name = "gpio2", |
| 1673 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1672 | .mpu_irqs = omap242x_gpio2_irqs, | 1674 | .mpu_irqs = omap242x_gpio2_irqs, |
| 1673 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio2_irqs), | 1675 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio2_irqs), |
| 1674 | .main_clk = "gpios_fck", | 1676 | .main_clk = "gpios_fck", |
| @@ -1699,6 +1701,7 @@ static struct omap_hwmod_ocp_if *omap2420_gpio3_slaves[] = { | |||
| 1699 | 1701 | ||
| 1700 | static struct omap_hwmod omap2420_gpio3_hwmod = { | 1702 | static struct omap_hwmod omap2420_gpio3_hwmod = { |
| 1701 | .name = "gpio3", | 1703 | .name = "gpio3", |
| 1704 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1702 | .mpu_irqs = omap242x_gpio3_irqs, | 1705 | .mpu_irqs = omap242x_gpio3_irqs, |
| 1703 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio3_irqs), | 1706 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio3_irqs), |
| 1704 | .main_clk = "gpios_fck", | 1707 | .main_clk = "gpios_fck", |
| @@ -1729,6 +1732,7 @@ static struct omap_hwmod_ocp_if *omap2420_gpio4_slaves[] = { | |||
| 1729 | 1732 | ||
| 1730 | static struct omap_hwmod omap2420_gpio4_hwmod = { | 1733 | static struct omap_hwmod omap2420_gpio4_hwmod = { |
| 1731 | .name = "gpio4", | 1734 | .name = "gpio4", |
| 1735 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1732 | .mpu_irqs = omap242x_gpio4_irqs, | 1736 | .mpu_irqs = omap242x_gpio4_irqs, |
| 1733 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio4_irqs), | 1737 | .mpu_irqs_cnt = ARRAY_SIZE(omap242x_gpio4_irqs), |
| 1734 | .main_clk = "gpios_fck", | 1738 | .main_clk = "gpios_fck", |
| @@ -1782,7 +1786,7 @@ static struct omap_hwmod_irq_info omap2420_dma_system_irqs[] = { | |||
| 1782 | static struct omap_hwmod_addr_space omap2420_dma_system_addrs[] = { | 1786 | static struct omap_hwmod_addr_space omap2420_dma_system_addrs[] = { |
| 1783 | { | 1787 | { |
| 1784 | .pa_start = 0x48056000, | 1788 | .pa_start = 0x48056000, |
| 1785 | .pa_end = 0x4a0560ff, | 1789 | .pa_end = 0x48056fff, |
| 1786 | .flags = ADDR_TYPE_RT | 1790 | .flags = ADDR_TYPE_RT |
| 1787 | }, | 1791 | }, |
| 1788 | }; | 1792 | }; |
diff --git a/arch/arm/mach-omap2/omap_hwmod_2430_data.c b/arch/arm/mach-omap2/omap_hwmod_2430_data.c index e6e3810db77f..9682dd519f8d 100644 --- a/arch/arm/mach-omap2/omap_hwmod_2430_data.c +++ b/arch/arm/mach-omap2/omap_hwmod_2430_data.c | |||
| @@ -1742,6 +1742,7 @@ static struct omap_hwmod_ocp_if *omap2430_gpio1_slaves[] = { | |||
| 1742 | 1742 | ||
| 1743 | static struct omap_hwmod omap2430_gpio1_hwmod = { | 1743 | static struct omap_hwmod omap2430_gpio1_hwmod = { |
| 1744 | .name = "gpio1", | 1744 | .name = "gpio1", |
| 1745 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1745 | .mpu_irqs = omap243x_gpio1_irqs, | 1746 | .mpu_irqs = omap243x_gpio1_irqs, |
| 1746 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio1_irqs), | 1747 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio1_irqs), |
| 1747 | .main_clk = "gpios_fck", | 1748 | .main_clk = "gpios_fck", |
| @@ -1772,6 +1773,7 @@ static struct omap_hwmod_ocp_if *omap2430_gpio2_slaves[] = { | |||
| 1772 | 1773 | ||
| 1773 | static struct omap_hwmod omap2430_gpio2_hwmod = { | 1774 | static struct omap_hwmod omap2430_gpio2_hwmod = { |
| 1774 | .name = "gpio2", | 1775 | .name = "gpio2", |
| 1776 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1775 | .mpu_irqs = omap243x_gpio2_irqs, | 1777 | .mpu_irqs = omap243x_gpio2_irqs, |
| 1776 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio2_irqs), | 1778 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio2_irqs), |
| 1777 | .main_clk = "gpios_fck", | 1779 | .main_clk = "gpios_fck", |
| @@ -1802,6 +1804,7 @@ static struct omap_hwmod_ocp_if *omap2430_gpio3_slaves[] = { | |||
| 1802 | 1804 | ||
| 1803 | static struct omap_hwmod omap2430_gpio3_hwmod = { | 1805 | static struct omap_hwmod omap2430_gpio3_hwmod = { |
| 1804 | .name = "gpio3", | 1806 | .name = "gpio3", |
| 1807 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1805 | .mpu_irqs = omap243x_gpio3_irqs, | 1808 | .mpu_irqs = omap243x_gpio3_irqs, |
| 1806 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio3_irqs), | 1809 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio3_irqs), |
| 1807 | .main_clk = "gpios_fck", | 1810 | .main_clk = "gpios_fck", |
| @@ -1832,6 +1835,7 @@ static struct omap_hwmod_ocp_if *omap2430_gpio4_slaves[] = { | |||
| 1832 | 1835 | ||
| 1833 | static struct omap_hwmod omap2430_gpio4_hwmod = { | 1836 | static struct omap_hwmod omap2430_gpio4_hwmod = { |
| 1834 | .name = "gpio4", | 1837 | .name = "gpio4", |
| 1838 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1835 | .mpu_irqs = omap243x_gpio4_irqs, | 1839 | .mpu_irqs = omap243x_gpio4_irqs, |
| 1836 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio4_irqs), | 1840 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio4_irqs), |
| 1837 | .main_clk = "gpios_fck", | 1841 | .main_clk = "gpios_fck", |
| @@ -1862,6 +1866,7 @@ static struct omap_hwmod_ocp_if *omap2430_gpio5_slaves[] = { | |||
| 1862 | 1866 | ||
| 1863 | static struct omap_hwmod omap2430_gpio5_hwmod = { | 1867 | static struct omap_hwmod omap2430_gpio5_hwmod = { |
| 1864 | .name = "gpio5", | 1868 | .name = "gpio5", |
| 1869 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 1865 | .mpu_irqs = omap243x_gpio5_irqs, | 1870 | .mpu_irqs = omap243x_gpio5_irqs, |
| 1866 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio5_irqs), | 1871 | .mpu_irqs_cnt = ARRAY_SIZE(omap243x_gpio5_irqs), |
| 1867 | .main_clk = "gpio5_fck", | 1872 | .main_clk = "gpio5_fck", |
| @@ -1915,7 +1920,7 @@ static struct omap_hwmod_irq_info omap2430_dma_system_irqs[] = { | |||
| 1915 | static struct omap_hwmod_addr_space omap2430_dma_system_addrs[] = { | 1920 | static struct omap_hwmod_addr_space omap2430_dma_system_addrs[] = { |
| 1916 | { | 1921 | { |
| 1917 | .pa_start = 0x48056000, | 1922 | .pa_start = 0x48056000, |
| 1918 | .pa_end = 0x4a0560ff, | 1923 | .pa_end = 0x48056fff, |
| 1919 | .flags = ADDR_TYPE_RT | 1924 | .flags = ADDR_TYPE_RT |
| 1920 | }, | 1925 | }, |
| 1921 | }; | 1926 | }; |
diff --git a/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c b/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c index b98e2dfcba28..909a84de6682 100644 --- a/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c +++ b/arch/arm/mach-omap2/omap_hwmod_3xxx_data.c | |||
| @@ -2141,6 +2141,7 @@ static struct omap_hwmod_ocp_if *omap3xxx_gpio1_slaves[] = { | |||
| 2141 | 2141 | ||
| 2142 | static struct omap_hwmod omap3xxx_gpio1_hwmod = { | 2142 | static struct omap_hwmod omap3xxx_gpio1_hwmod = { |
| 2143 | .name = "gpio1", | 2143 | .name = "gpio1", |
| 2144 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 2144 | .mpu_irqs = omap3xxx_gpio1_irqs, | 2145 | .mpu_irqs = omap3xxx_gpio1_irqs, |
| 2145 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio1_irqs), | 2146 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio1_irqs), |
| 2146 | .main_clk = "gpio1_ick", | 2147 | .main_clk = "gpio1_ick", |
| @@ -2177,6 +2178,7 @@ static struct omap_hwmod_ocp_if *omap3xxx_gpio2_slaves[] = { | |||
| 2177 | 2178 | ||
| 2178 | static struct omap_hwmod omap3xxx_gpio2_hwmod = { | 2179 | static struct omap_hwmod omap3xxx_gpio2_hwmod = { |
| 2179 | .name = "gpio2", | 2180 | .name = "gpio2", |
| 2181 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 2180 | .mpu_irqs = omap3xxx_gpio2_irqs, | 2182 | .mpu_irqs = omap3xxx_gpio2_irqs, |
| 2181 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio2_irqs), | 2183 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio2_irqs), |
| 2182 | .main_clk = "gpio2_ick", | 2184 | .main_clk = "gpio2_ick", |
| @@ -2213,6 +2215,7 @@ static struct omap_hwmod_ocp_if *omap3xxx_gpio3_slaves[] = { | |||
| 2213 | 2215 | ||
| 2214 | static struct omap_hwmod omap3xxx_gpio3_hwmod = { | 2216 | static struct omap_hwmod omap3xxx_gpio3_hwmod = { |
| 2215 | .name = "gpio3", | 2217 | .name = "gpio3", |
| 2218 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 2216 | .mpu_irqs = omap3xxx_gpio3_irqs, | 2219 | .mpu_irqs = omap3xxx_gpio3_irqs, |
| 2217 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio3_irqs), | 2220 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio3_irqs), |
| 2218 | .main_clk = "gpio3_ick", | 2221 | .main_clk = "gpio3_ick", |
| @@ -2249,6 +2252,7 @@ static struct omap_hwmod_ocp_if *omap3xxx_gpio4_slaves[] = { | |||
| 2249 | 2252 | ||
| 2250 | static struct omap_hwmod omap3xxx_gpio4_hwmod = { | 2253 | static struct omap_hwmod omap3xxx_gpio4_hwmod = { |
| 2251 | .name = "gpio4", | 2254 | .name = "gpio4", |
| 2255 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 2252 | .mpu_irqs = omap3xxx_gpio4_irqs, | 2256 | .mpu_irqs = omap3xxx_gpio4_irqs, |
| 2253 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio4_irqs), | 2257 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio4_irqs), |
| 2254 | .main_clk = "gpio4_ick", | 2258 | .main_clk = "gpio4_ick", |
| @@ -2285,6 +2289,7 @@ static struct omap_hwmod_ocp_if *omap3xxx_gpio5_slaves[] = { | |||
| 2285 | 2289 | ||
| 2286 | static struct omap_hwmod omap3xxx_gpio5_hwmod = { | 2290 | static struct omap_hwmod omap3xxx_gpio5_hwmod = { |
| 2287 | .name = "gpio5", | 2291 | .name = "gpio5", |
| 2292 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 2288 | .mpu_irqs = omap3xxx_gpio5_irqs, | 2293 | .mpu_irqs = omap3xxx_gpio5_irqs, |
| 2289 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio5_irqs), | 2294 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio5_irqs), |
| 2290 | .main_clk = "gpio5_ick", | 2295 | .main_clk = "gpio5_ick", |
| @@ -2321,6 +2326,7 @@ static struct omap_hwmod_ocp_if *omap3xxx_gpio6_slaves[] = { | |||
| 2321 | 2326 | ||
| 2322 | static struct omap_hwmod omap3xxx_gpio6_hwmod = { | 2327 | static struct omap_hwmod omap3xxx_gpio6_hwmod = { |
| 2323 | .name = "gpio6", | 2328 | .name = "gpio6", |
| 2329 | .flags = HWMOD_CONTROL_OPT_CLKS_IN_RESET, | ||
| 2324 | .mpu_irqs = omap3xxx_gpio6_irqs, | 2330 | .mpu_irqs = omap3xxx_gpio6_irqs, |
| 2325 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio6_irqs), | 2331 | .mpu_irqs_cnt = ARRAY_SIZE(omap3xxx_gpio6_irqs), |
| 2326 | .main_clk = "gpio6_ick", | 2332 | .main_clk = "gpio6_ick", |
| @@ -2386,7 +2392,7 @@ static struct omap_hwmod_irq_info omap3xxx_dma_system_irqs[] = { | |||
| 2386 | static struct omap_hwmod_addr_space omap3xxx_dma_system_addrs[] = { | 2392 | static struct omap_hwmod_addr_space omap3xxx_dma_system_addrs[] = { |
| 2387 | { | 2393 | { |
| 2388 | .pa_start = 0x48056000, | 2394 | .pa_start = 0x48056000, |
| 2389 | .pa_end = 0x4a0560ff, | 2395 | .pa_end = 0x48056fff, |
| 2390 | .flags = ADDR_TYPE_RT | 2396 | .flags = ADDR_TYPE_RT |
| 2391 | }, | 2397 | }, |
| 2392 | }; | 2398 | }; |
diff --git a/arch/arm/mach-omap2/omap_hwmod_44xx_data.c b/arch/arm/mach-omap2/omap_hwmod_44xx_data.c index 3e88dd3f8ef3..abc548a0c98d 100644 --- a/arch/arm/mach-omap2/omap_hwmod_44xx_data.c +++ b/arch/arm/mach-omap2/omap_hwmod_44xx_data.c | |||
| @@ -885,7 +885,7 @@ static struct omap_hwmod_ocp_if *omap44xx_dma_system_masters[] = { | |||
| 885 | static struct omap_hwmod_addr_space omap44xx_dma_system_addrs[] = { | 885 | static struct omap_hwmod_addr_space omap44xx_dma_system_addrs[] = { |
| 886 | { | 886 | { |
| 887 | .pa_start = 0x4a056000, | 887 | .pa_start = 0x4a056000, |
| 888 | .pa_end = 0x4a0560ff, | 888 | .pa_end = 0x4a056fff, |
| 889 | .flags = ADDR_TYPE_RT | 889 | .flags = ADDR_TYPE_RT |
| 890 | }, | 890 | }, |
| 891 | }; | 891 | }; |
diff --git a/arch/arm/mach-omap2/omap_l3_smx.c b/arch/arm/mach-omap2/omap_l3_smx.c index 5f2da7565b68..4321e7938929 100644 --- a/arch/arm/mach-omap2/omap_l3_smx.c +++ b/arch/arm/mach-omap2/omap_l3_smx.c | |||
| @@ -196,11 +196,11 @@ static irqreturn_t omap3_l3_app_irq(int irq, void *_l3) | |||
| 196 | /* No timeout error for debug sources */ | 196 | /* No timeout error for debug sources */ |
| 197 | } | 197 | } |
| 198 | 198 | ||
| 199 | base = ((l3->rt) + (*(omap3_l3_bases[int_type] + err_source))); | ||
| 200 | |||
| 201 | /* identify the error source */ | 199 | /* identify the error source */ |
| 202 | for (err_source = 0; !(status & (1 << err_source)); err_source++) | 200 | for (err_source = 0; !(status & (1 << err_source)); err_source++) |
| 203 | ; | 201 | ; |
| 202 | |||
| 203 | base = l3->rt + *(omap3_l3_bases[int_type] + err_source); | ||
| 204 | error = omap3_l3_readll(base, L3_ERROR_LOG); | 204 | error = omap3_l3_readll(base, L3_ERROR_LOG); |
| 205 | 205 | ||
| 206 | if (error) { | 206 | if (error) { |
diff --git a/arch/arm/mach-omap2/pm.c b/arch/arm/mach-omap2/pm.c index 30af3351c2d6..49486f522dca 100644 --- a/arch/arm/mach-omap2/pm.c +++ b/arch/arm/mach-omap2/pm.c | |||
| @@ -89,6 +89,7 @@ static void omap2_init_processor_devices(void) | |||
| 89 | if (cpu_is_omap44xx()) { | 89 | if (cpu_is_omap44xx()) { |
| 90 | _init_omap_device("l3_main_1", &l3_dev); | 90 | _init_omap_device("l3_main_1", &l3_dev); |
| 91 | _init_omap_device("dsp", &dsp_dev); | 91 | _init_omap_device("dsp", &dsp_dev); |
| 92 | _init_omap_device("iva", &iva_dev); | ||
| 92 | } else { | 93 | } else { |
| 93 | _init_omap_device("l3_main", &l3_dev); | 94 | _init_omap_device("l3_main", &l3_dev); |
| 94 | } | 95 | } |
diff --git a/arch/arm/mach-omap2/voltage.c b/arch/arm/mach-omap2/voltage.c index 6fb520999b6e..0c1552d9d995 100644 --- a/arch/arm/mach-omap2/voltage.c +++ b/arch/arm/mach-omap2/voltage.c | |||
| @@ -114,7 +114,6 @@ static int __init _config_common_vdd_data(struct omap_vdd_info *vdd) | |||
| 114 | sys_clk_speed /= 1000; | 114 | sys_clk_speed /= 1000; |
| 115 | 115 | ||
| 116 | /* Generic voltage parameters */ | 116 | /* Generic voltage parameters */ |
| 117 | vdd->curr_volt = 1200000; | ||
| 118 | vdd->volt_scale = vp_forceupdate_scale_voltage; | 117 | vdd->volt_scale = vp_forceupdate_scale_voltage; |
| 119 | vdd->vp_enabled = false; | 118 | vdd->vp_enabled = false; |
| 120 | 119 | ||
diff --git a/arch/arm/mach-pxa/hx4700.c b/arch/arm/mach-pxa/hx4700.c index 6de0ad0eea65..9cdcca597924 100644 --- a/arch/arm/mach-pxa/hx4700.c +++ b/arch/arm/mach-pxa/hx4700.c | |||
| @@ -711,7 +711,7 @@ static struct regulator_consumer_supply bq24022_consumers[] = { | |||
| 711 | static struct regulator_init_data bq24022_init_data = { | 711 | static struct regulator_init_data bq24022_init_data = { |
| 712 | .constraints = { | 712 | .constraints = { |
| 713 | .max_uA = 500000, | 713 | .max_uA = 500000, |
| 714 | .valid_ops_mask = REGULATOR_CHANGE_CURRENT, | 714 | .valid_ops_mask = REGULATOR_CHANGE_CURRENT|REGULATOR_CHANGE_STATUS, |
| 715 | }, | 715 | }, |
| 716 | .num_consumer_supplies = ARRAY_SIZE(bq24022_consumers), | 716 | .num_consumer_supplies = ARRAY_SIZE(bq24022_consumers), |
| 717 | .consumer_supplies = bq24022_consumers, | 717 | .consumer_supplies = bq24022_consumers, |
diff --git a/arch/arm/mach-pxa/magician.c b/arch/arm/mach-pxa/magician.c index a72993dde2b3..9984ef70bd79 100644 --- a/arch/arm/mach-pxa/magician.c +++ b/arch/arm/mach-pxa/magician.c | |||
| @@ -599,7 +599,7 @@ static struct regulator_consumer_supply bq24022_consumers[] = { | |||
| 599 | static struct regulator_init_data bq24022_init_data = { | 599 | static struct regulator_init_data bq24022_init_data = { |
| 600 | .constraints = { | 600 | .constraints = { |
| 601 | .max_uA = 500000, | 601 | .max_uA = 500000, |
| 602 | .valid_ops_mask = REGULATOR_CHANGE_CURRENT, | 602 | .valid_ops_mask = REGULATOR_CHANGE_CURRENT | REGULATOR_CHANGE_STATUS, |
| 603 | }, | 603 | }, |
| 604 | .num_consumer_supplies = ARRAY_SIZE(bq24022_consumers), | 604 | .num_consumer_supplies = ARRAY_SIZE(bq24022_consumers), |
| 605 | .consumer_supplies = bq24022_consumers, | 605 | .consumer_supplies = bq24022_consumers, |
diff --git a/arch/arm/mm/proc-xscale.S b/arch/arm/mm/proc-xscale.S index ce233bcbf506..42af97664c9d 100644 --- a/arch/arm/mm/proc-xscale.S +++ b/arch/arm/mm/proc-xscale.S | |||
| @@ -395,7 +395,7 @@ ENTRY(xscale_dma_a0_map_area) | |||
| 395 | teq r2, #DMA_TO_DEVICE | 395 | teq r2, #DMA_TO_DEVICE |
| 396 | beq xscale_dma_clean_range | 396 | beq xscale_dma_clean_range |
| 397 | b xscale_dma_flush_range | 397 | b xscale_dma_flush_range |
| 398 | ENDPROC(xscsale_dma_a0_map_area) | 398 | ENDPROC(xscale_dma_a0_map_area) |
| 399 | 399 | ||
| 400 | /* | 400 | /* |
| 401 | * dma_unmap_area(start, size, dir) | 401 | * dma_unmap_area(start, size, dir) |
diff --git a/arch/arm/plat-mxc/gpio.c b/arch/arm/plat-mxc/gpio.c index 7a107246fd98..6cd6d7f686f6 100644 --- a/arch/arm/plat-mxc/gpio.c +++ b/arch/arm/plat-mxc/gpio.c | |||
| @@ -295,6 +295,12 @@ static int mxc_gpio_direction_output(struct gpio_chip *chip, | |||
| 295 | return 0; | 295 | return 0; |
| 296 | } | 296 | } |
| 297 | 297 | ||
| 298 | /* | ||
| 299 | * This lock class tells lockdep that GPIO irqs are in a different | ||
| 300 | * category than their parents, so it won't report false recursion. | ||
| 301 | */ | ||
| 302 | static struct lock_class_key gpio_lock_class; | ||
| 303 | |||
| 298 | int __init mxc_gpio_init(struct mxc_gpio_port *port, int cnt) | 304 | int __init mxc_gpio_init(struct mxc_gpio_port *port, int cnt) |
| 299 | { | 305 | { |
| 300 | int i, j; | 306 | int i, j; |
| @@ -311,6 +317,7 @@ int __init mxc_gpio_init(struct mxc_gpio_port *port, int cnt) | |||
| 311 | __raw_writel(~0, port[i].base + GPIO_ISR); | 317 | __raw_writel(~0, port[i].base + GPIO_ISR); |
| 312 | for (j = port[i].virtual_irq_start; | 318 | for (j = port[i].virtual_irq_start; |
| 313 | j < port[i].virtual_irq_start + 32; j++) { | 319 | j < port[i].virtual_irq_start + 32; j++) { |
| 320 | irq_set_lockdep_class(j, &gpio_lock_class); | ||
| 314 | irq_set_chip_and_handler(j, &gpio_irq_chip, | 321 | irq_set_chip_and_handler(j, &gpio_irq_chip, |
| 315 | handle_level_irq); | 322 | handle_level_irq); |
| 316 | set_irq_flags(j, IRQF_VALID); | 323 | set_irq_flags(j, IRQF_VALID); |
diff --git a/arch/arm/plat-mxc/ssi-fiq.S b/arch/arm/plat-mxc/ssi-fiq.S index 4ddce565b353..8397a2dd19f2 100644 --- a/arch/arm/plat-mxc/ssi-fiq.S +++ b/arch/arm/plat-mxc/ssi-fiq.S | |||
| @@ -124,6 +124,8 @@ imx_ssi_fiq_start: | |||
| 124 | 1: | 124 | 1: |
| 125 | @ return from FIQ | 125 | @ return from FIQ |
| 126 | subs pc, lr, #4 | 126 | subs pc, lr, #4 |
| 127 | |||
| 128 | .align | ||
| 127 | imx_ssi_fiq_base: | 129 | imx_ssi_fiq_base: |
| 128 | .word 0x0 | 130 | .word 0x0 |
| 129 | imx_ssi_fiq_rx_buffer: | 131 | imx_ssi_fiq_rx_buffer: |
diff --git a/arch/m68k/mm/motorola.c b/arch/m68k/mm/motorola.c index 02b7a03e4226..8b3db1c587fc 100644 --- a/arch/m68k/mm/motorola.c +++ b/arch/m68k/mm/motorola.c | |||
| @@ -300,6 +300,8 @@ void __init paging_init(void) | |||
| 300 | zones_size[ZONE_DMA] = m68k_memory[i].size >> PAGE_SHIFT; | 300 | zones_size[ZONE_DMA] = m68k_memory[i].size >> PAGE_SHIFT; |
| 301 | free_area_init_node(i, zones_size, | 301 | free_area_init_node(i, zones_size, |
| 302 | m68k_memory[i].addr >> PAGE_SHIFT, NULL); | 302 | m68k_memory[i].addr >> PAGE_SHIFT, NULL); |
| 303 | if (node_present_pages(i)) | ||
| 304 | node_set_state(i, N_NORMAL_MEMORY); | ||
| 303 | } | 305 | } |
| 304 | } | 306 | } |
| 305 | 307 | ||
diff --git a/arch/parisc/mm/init.c b/arch/parisc/mm/init.c index b7ed8d7a9b33..b1d126258dee 100644 --- a/arch/parisc/mm/init.c +++ b/arch/parisc/mm/init.c | |||
| @@ -266,8 +266,10 @@ static void __init setup_bootmem(void) | |||
| 266 | } | 266 | } |
| 267 | memset(pfnnid_map, 0xff, sizeof(pfnnid_map)); | 267 | memset(pfnnid_map, 0xff, sizeof(pfnnid_map)); |
| 268 | 268 | ||
| 269 | for (i = 0; i < npmem_ranges; i++) | 269 | for (i = 0; i < npmem_ranges; i++) { |
| 270 | node_set_state(i, N_NORMAL_MEMORY); | ||
| 270 | node_set_online(i); | 271 | node_set_online(i); |
| 272 | } | ||
| 271 | #endif | 273 | #endif |
| 272 | 274 | ||
| 273 | /* | 275 | /* |
diff --git a/arch/powerpc/include/asm/8xx_immap.h b/arch/powerpc/include/asm/8xx_immap.h index 6b6dc20b0beb..bdf0563ba423 100644 --- a/arch/powerpc/include/asm/8xx_immap.h +++ b/arch/powerpc/include/asm/8xx_immap.h | |||
| @@ -393,8 +393,8 @@ typedef struct fec { | |||
| 393 | uint fec_addr_low; /* lower 32 bits of station address */ | 393 | uint fec_addr_low; /* lower 32 bits of station address */ |
| 394 | ushort fec_addr_high; /* upper 16 bits of station address */ | 394 | ushort fec_addr_high; /* upper 16 bits of station address */ |
| 395 | ushort res1; /* reserved */ | 395 | ushort res1; /* reserved */ |
| 396 | uint fec_hash_table_high; /* upper 32-bits of hash table */ | 396 | uint fec_grp_hash_table_high; /* upper 32-bits of hash table */ |
| 397 | uint fec_hash_table_low; /* lower 32-bits of hash table */ | 397 | uint fec_grp_hash_table_low; /* lower 32-bits of hash table */ |
| 398 | uint fec_r_des_start; /* beginning of Rx descriptor ring */ | 398 | uint fec_r_des_start; /* beginning of Rx descriptor ring */ |
| 399 | uint fec_x_des_start; /* beginning of Tx descriptor ring */ | 399 | uint fec_x_des_start; /* beginning of Tx descriptor ring */ |
| 400 | uint fec_r_buff_size; /* Rx buffer size */ | 400 | uint fec_r_buff_size; /* Rx buffer size */ |
diff --git a/arch/powerpc/include/asm/uninorth.h b/arch/powerpc/include/asm/uninorth.h index ae9c899c8a6d..d12b11d7641e 100644 --- a/arch/powerpc/include/asm/uninorth.h +++ b/arch/powerpc/include/asm/uninorth.h | |||
| @@ -60,7 +60,7 @@ | |||
| 60 | * | 60 | * |
| 61 | * Obviously, the GART is not cache coherent and so any change to it | 61 | * Obviously, the GART is not cache coherent and so any change to it |
| 62 | * must be flushed to memory (or maybe just make the GART space non | 62 | * must be flushed to memory (or maybe just make the GART space non |
| 63 | * cachable). AGP memory itself does't seem to be cache coherent neither. | 63 | * cachable). AGP memory itself doesn't seem to be cache coherent neither. |
| 64 | * | 64 | * |
| 65 | * In order to invalidate the GART (which is probably necessary to inval | 65 | * In order to invalidate the GART (which is probably necessary to inval |
| 66 | * the bridge internal TLBs), the following sequence has to be written, | 66 | * the bridge internal TLBs), the following sequence has to be written, |
diff --git a/arch/s390/crypto/prng.c b/arch/s390/crypto/prng.c index 44bca3f994b0..8b16c479585b 100644 --- a/arch/s390/crypto/prng.c +++ b/arch/s390/crypto/prng.c | |||
| @@ -76,7 +76,7 @@ static void prng_seed(int nbytes) | |||
| 76 | 76 | ||
| 77 | /* Add the entropy */ | 77 | /* Add the entropy */ |
| 78 | while (nbytes >= 8) { | 78 | while (nbytes >= 8) { |
| 79 | *((__u64 *)parm_block) ^= *((__u64 *)buf+i); | 79 | *((__u64 *)parm_block) ^= *((__u64 *)(buf+i)); |
| 80 | prng_add_entropy(); | 80 | prng_add_entropy(); |
| 81 | i += 8; | 81 | i += 8; |
| 82 | nbytes -= 8; | 82 | nbytes -= 8; |
diff --git a/arch/s390/mm/fault.c b/arch/s390/mm/fault.c index 4cf85fef407c..ab988135e5c6 100644 --- a/arch/s390/mm/fault.c +++ b/arch/s390/mm/fault.c | |||
| @@ -543,7 +543,6 @@ static void pfault_interrupt(unsigned int ext_int_code, | |||
| 543 | struct task_struct *tsk; | 543 | struct task_struct *tsk; |
| 544 | __u16 subcode; | 544 | __u16 subcode; |
| 545 | 545 | ||
| 546 | kstat_cpu(smp_processor_id()).irqs[EXTINT_PFL]++; | ||
| 547 | /* | 546 | /* |
| 548 | * Get the external interruption subcode & pfault | 547 | * Get the external interruption subcode & pfault |
| 549 | * initial/completion signal bit. VM stores this | 548 | * initial/completion signal bit. VM stores this |
| @@ -553,6 +552,7 @@ static void pfault_interrupt(unsigned int ext_int_code, | |||
| 553 | subcode = ext_int_code >> 16; | 552 | subcode = ext_int_code >> 16; |
| 554 | if ((subcode & 0xff00) != __SUBCODE_MASK) | 553 | if ((subcode & 0xff00) != __SUBCODE_MASK) |
| 555 | return; | 554 | return; |
| 555 | kstat_cpu(smp_processor_id()).irqs[EXTINT_PFL]++; | ||
| 556 | 556 | ||
| 557 | /* | 557 | /* |
| 558 | * Get the token (= address of the task structure of the affected task). | 558 | * Get the token (= address of the task structure of the affected task). |
diff --git a/arch/um/Kconfig.um b/arch/um/Kconfig.um index 90a438acbfaf..b5e675e370c6 100644 --- a/arch/um/Kconfig.um +++ b/arch/um/Kconfig.um | |||
| @@ -47,7 +47,7 @@ config HOSTFS | |||
| 47 | 47 | ||
| 48 | config HPPFS | 48 | config HPPFS |
| 49 | tristate "HoneyPot ProcFS (EXPERIMENTAL)" | 49 | tristate "HoneyPot ProcFS (EXPERIMENTAL)" |
| 50 | depends on EXPERIMENTAL | 50 | depends on EXPERIMENTAL && PROC_FS |
| 51 | help | 51 | help |
| 52 | hppfs (HoneyPot ProcFS) is a filesystem which allows UML /proc | 52 | hppfs (HoneyPot ProcFS) is a filesystem which allows UML /proc |
| 53 | entries to be overridden, removed, or fabricated from the host. | 53 | entries to be overridden, removed, or fabricated from the host. |
diff --git a/arch/um/include/asm/thread_info.h b/arch/um/include/asm/thread_info.h index e2cf786bda0a..5bd1bad33fab 100644 --- a/arch/um/include/asm/thread_info.h +++ b/arch/um/include/asm/thread_info.h | |||
| @@ -49,7 +49,10 @@ static inline struct thread_info *current_thread_info(void) | |||
| 49 | { | 49 | { |
| 50 | struct thread_info *ti; | 50 | struct thread_info *ti; |
| 51 | unsigned long mask = THREAD_SIZE - 1; | 51 | unsigned long mask = THREAD_SIZE - 1; |
| 52 | ti = (struct thread_info *) (((unsigned long) &ti) & ~mask); | 52 | void *p; |
| 53 | |||
| 54 | asm volatile ("" : "=r" (p) : "0" (&ti)); | ||
| 55 | ti = (struct thread_info *) (((unsigned long)p) & ~mask); | ||
| 53 | return ti; | 56 | return ti; |
| 54 | } | 57 | } |
| 55 | 58 | ||
diff --git a/arch/um/sys-i386/Makefile b/arch/um/sys-i386/Makefile index 804b28dd0328..b1da91c1b200 100644 --- a/arch/um/sys-i386/Makefile +++ b/arch/um/sys-i386/Makefile | |||
| @@ -4,7 +4,7 @@ | |||
| 4 | 4 | ||
| 5 | obj-y = bug.o bugs.o checksum.o delay.o fault.o ksyms.o ldt.o ptrace.o \ | 5 | obj-y = bug.o bugs.o checksum.o delay.o fault.o ksyms.o ldt.o ptrace.o \ |
| 6 | ptrace_user.o setjmp.o signal.o stub.o stub_segv.o syscalls.o sysrq.o \ | 6 | ptrace_user.o setjmp.o signal.o stub.o stub_segv.o syscalls.o sysrq.o \ |
| 7 | sys_call_table.o tls.o | 7 | sys_call_table.o tls.o atomic64_cx8_32.o |
| 8 | 8 | ||
| 9 | obj-$(CONFIG_BINFMT_ELF) += elfcore.o | 9 | obj-$(CONFIG_BINFMT_ELF) += elfcore.o |
| 10 | 10 | ||
diff --git a/arch/um/sys-i386/atomic64_cx8_32.S b/arch/um/sys-i386/atomic64_cx8_32.S new file mode 100644 index 000000000000..1e901d3d4a95 --- /dev/null +++ b/arch/um/sys-i386/atomic64_cx8_32.S | |||
| @@ -0,0 +1,225 @@ | |||
| 1 | /* | ||
| 2 | * atomic64_t for 586+ | ||
| 3 | * | ||
| 4 | * Copied from arch/x86/lib/atomic64_cx8_32.S | ||
| 5 | * | ||
| 6 | * Copyright © 2010 Luca Barbieri | ||
| 7 | * | ||
| 8 | * This program is free software; you can redistribute it and/or modify | ||
| 9 | * it under the terms of the GNU General Public License as published by | ||
| 10 | * the Free Software Foundation; either version 2 of the License, or | ||
| 11 | * (at your option) any later version. | ||
| 12 | * | ||
| 13 | */ | ||
| 14 | |||
| 15 | #include <linux/linkage.h> | ||
| 16 | #include <asm/alternative-asm.h> | ||
| 17 | #include <asm/dwarf2.h> | ||
| 18 | |||
| 19 | .macro SAVE reg | ||
| 20 | pushl_cfi %\reg | ||
| 21 | CFI_REL_OFFSET \reg, 0 | ||
| 22 | .endm | ||
| 23 | |||
| 24 | .macro RESTORE reg | ||
| 25 | popl_cfi %\reg | ||
| 26 | CFI_RESTORE \reg | ||
| 27 | .endm | ||
| 28 | |||
| 29 | .macro read64 reg | ||
| 30 | movl %ebx, %eax | ||
| 31 | movl %ecx, %edx | ||
| 32 | /* we need LOCK_PREFIX since otherwise cmpxchg8b always does the write */ | ||
| 33 | LOCK_PREFIX | ||
| 34 | cmpxchg8b (\reg) | ||
| 35 | .endm | ||
| 36 | |||
| 37 | ENTRY(atomic64_read_cx8) | ||
| 38 | CFI_STARTPROC | ||
| 39 | |||
| 40 | read64 %ecx | ||
| 41 | ret | ||
| 42 | CFI_ENDPROC | ||
| 43 | ENDPROC(atomic64_read_cx8) | ||
| 44 | |||
| 45 | ENTRY(atomic64_set_cx8) | ||
| 46 | CFI_STARTPROC | ||
| 47 | |||
| 48 | 1: | ||
| 49 | /* we don't need LOCK_PREFIX since aligned 64-bit writes | ||
| 50 | * are atomic on 586 and newer */ | ||
| 51 | cmpxchg8b (%esi) | ||
| 52 | jne 1b | ||
| 53 | |||
| 54 | ret | ||
| 55 | CFI_ENDPROC | ||
| 56 | ENDPROC(atomic64_set_cx8) | ||
| 57 | |||
| 58 | ENTRY(atomic64_xchg_cx8) | ||
| 59 | CFI_STARTPROC | ||
| 60 | |||
| 61 | movl %ebx, %eax | ||
| 62 | movl %ecx, %edx | ||
| 63 | 1: | ||
| 64 | LOCK_PREFIX | ||
| 65 | cmpxchg8b (%esi) | ||
| 66 | jne 1b | ||
| 67 | |||
| 68 | ret | ||
| 69 | CFI_ENDPROC | ||
| 70 | ENDPROC(atomic64_xchg_cx8) | ||
| 71 | |||
| 72 | .macro addsub_return func ins insc | ||
| 73 | ENTRY(atomic64_\func\()_return_cx8) | ||
| 74 | CFI_STARTPROC | ||
| 75 | SAVE ebp | ||
| 76 | SAVE ebx | ||
| 77 | SAVE esi | ||
| 78 | SAVE edi | ||
| 79 | |||
| 80 | movl %eax, %esi | ||
| 81 | movl %edx, %edi | ||
| 82 | movl %ecx, %ebp | ||
| 83 | |||
| 84 | read64 %ebp | ||
| 85 | 1: | ||
| 86 | movl %eax, %ebx | ||
| 87 | movl %edx, %ecx | ||
| 88 | \ins\()l %esi, %ebx | ||
| 89 | \insc\()l %edi, %ecx | ||
| 90 | LOCK_PREFIX | ||
| 91 | cmpxchg8b (%ebp) | ||
| 92 | jne 1b | ||
| 93 | |||
| 94 | 10: | ||
| 95 | movl %ebx, %eax | ||
| 96 | movl %ecx, %edx | ||
| 97 | RESTORE edi | ||
| 98 | RESTORE esi | ||
| 99 | RESTORE ebx | ||
| 100 | RESTORE ebp | ||
| 101 | ret | ||
| 102 | CFI_ENDPROC | ||
| 103 | ENDPROC(atomic64_\func\()_return_cx8) | ||
| 104 | .endm | ||
| 105 | |||
| 106 | addsub_return add add adc | ||
| 107 | addsub_return sub sub sbb | ||
| 108 | |||
| 109 | .macro incdec_return func ins insc | ||
| 110 | ENTRY(atomic64_\func\()_return_cx8) | ||
| 111 | CFI_STARTPROC | ||
| 112 | SAVE ebx | ||
| 113 | |||
| 114 | read64 %esi | ||
| 115 | 1: | ||
| 116 | movl %eax, %ebx | ||
| 117 | movl %edx, %ecx | ||
| 118 | \ins\()l $1, %ebx | ||
| 119 | \insc\()l $0, %ecx | ||
| 120 | LOCK_PREFIX | ||
| 121 | cmpxchg8b (%esi) | ||
| 122 | jne 1b | ||
| 123 | |||
| 124 | 10: | ||
| 125 | movl %ebx, %eax | ||
| 126 | movl %ecx, %edx | ||
| 127 | RESTORE ebx | ||
| 128 | ret | ||
| 129 | CFI_ENDPROC | ||
| 130 | ENDPROC(atomic64_\func\()_return_cx8) | ||
| 131 | .endm | ||
| 132 | |||
| 133 | incdec_return inc add adc | ||
| 134 | incdec_return dec sub sbb | ||
| 135 | |||
| 136 | ENTRY(atomic64_dec_if_positive_cx8) | ||
| 137 | CFI_STARTPROC | ||
| 138 | SAVE ebx | ||
| 139 | |||
| 140 | read64 %esi | ||
| 141 | 1: | ||
| 142 | movl %eax, %ebx | ||
| 143 | movl %edx, %ecx | ||
| 144 | subl $1, %ebx | ||
| 145 | sbb $0, %ecx | ||
| 146 | js 2f | ||
| 147 | LOCK_PREFIX | ||
| 148 | cmpxchg8b (%esi) | ||
| 149 | jne 1b | ||
| 150 | |||
| 151 | 2: | ||
| 152 | movl %ebx, %eax | ||
| 153 | movl %ecx, %edx | ||
| 154 | RESTORE ebx | ||
| 155 | ret | ||
| 156 | CFI_ENDPROC | ||
| 157 | ENDPROC(atomic64_dec_if_positive_cx8) | ||
| 158 | |||
| 159 | ENTRY(atomic64_add_unless_cx8) | ||
| 160 | CFI_STARTPROC | ||
| 161 | SAVE ebp | ||
| 162 | SAVE ebx | ||
| 163 | /* these just push these two parameters on the stack */ | ||
| 164 | SAVE edi | ||
| 165 | SAVE esi | ||
| 166 | |||
| 167 | movl %ecx, %ebp | ||
| 168 | movl %eax, %esi | ||
| 169 | movl %edx, %edi | ||
| 170 | |||
| 171 | read64 %ebp | ||
| 172 | 1: | ||
| 173 | cmpl %eax, 0(%esp) | ||
| 174 | je 4f | ||
| 175 | 2: | ||
| 176 | movl %eax, %ebx | ||
| 177 | movl %edx, %ecx | ||
| 178 | addl %esi, %ebx | ||
| 179 | adcl %edi, %ecx | ||
| 180 | LOCK_PREFIX | ||
| 181 | cmpxchg8b (%ebp) | ||
| 182 | jne 1b | ||
| 183 | |||
| 184 | movl $1, %eax | ||
| 185 | 3: | ||
| 186 | addl $8, %esp | ||
| 187 | CFI_ADJUST_CFA_OFFSET -8 | ||
| 188 | RESTORE ebx | ||
| 189 | RESTORE ebp | ||
| 190 | ret | ||
| 191 | 4: | ||
| 192 | cmpl %edx, 4(%esp) | ||
| 193 | jne 2b | ||
| 194 | xorl %eax, %eax | ||
| 195 | jmp 3b | ||
| 196 | CFI_ENDPROC | ||
| 197 | ENDPROC(atomic64_add_unless_cx8) | ||
| 198 | |||
| 199 | ENTRY(atomic64_inc_not_zero_cx8) | ||
| 200 | CFI_STARTPROC | ||
| 201 | SAVE ebx | ||
| 202 | |||
| 203 | read64 %esi | ||
| 204 | 1: | ||
| 205 | testl %eax, %eax | ||
| 206 | je 4f | ||
| 207 | 2: | ||
| 208 | movl %eax, %ebx | ||
| 209 | movl %edx, %ecx | ||
| 210 | addl $1, %ebx | ||
| 211 | adcl $0, %ecx | ||
| 212 | LOCK_PREFIX | ||
| 213 | cmpxchg8b (%esi) | ||
| 214 | jne 1b | ||
| 215 | |||
| 216 | movl $1, %eax | ||
| 217 | 3: | ||
| 218 | RESTORE ebx | ||
| 219 | ret | ||
| 220 | 4: | ||
| 221 | testl %edx, %edx | ||
| 222 | jne 2b | ||
| 223 | jmp 3b | ||
| 224 | CFI_ENDPROC | ||
| 225 | ENDPROC(atomic64_inc_not_zero_cx8) | ||
diff --git a/arch/x86/boot/memory.c b/arch/x86/boot/memory.c index cae3feb1035e..db75d07c3645 100644 --- a/arch/x86/boot/memory.c +++ b/arch/x86/boot/memory.c | |||
| @@ -91,7 +91,7 @@ static int detect_memory_e801(void) | |||
| 91 | if (oreg.ax > 15*1024) { | 91 | if (oreg.ax > 15*1024) { |
| 92 | return -1; /* Bogus! */ | 92 | return -1; /* Bogus! */ |
| 93 | } else if (oreg.ax == 15*1024) { | 93 | } else if (oreg.ax == 15*1024) { |
| 94 | boot_params.alt_mem_k = (oreg.dx << 6) + oreg.ax; | 94 | boot_params.alt_mem_k = (oreg.bx << 6) + oreg.ax; |
| 95 | } else { | 95 | } else { |
| 96 | /* | 96 | /* |
| 97 | * This ignores memory above 16MB if we have a memory | 97 | * This ignores memory above 16MB if we have a memory |
diff --git a/arch/x86/include/asm/io_apic.h b/arch/x86/include/asm/io_apic.h index c4bd267dfc50..a97a240f67f3 100644 --- a/arch/x86/include/asm/io_apic.h +++ b/arch/x86/include/asm/io_apic.h | |||
| @@ -150,7 +150,7 @@ void setup_IO_APIC_irq_extra(u32 gsi); | |||
| 150 | extern void ioapic_and_gsi_init(void); | 150 | extern void ioapic_and_gsi_init(void); |
| 151 | extern void ioapic_insert_resources(void); | 151 | extern void ioapic_insert_resources(void); |
| 152 | 152 | ||
| 153 | int io_apic_setup_irq_pin(unsigned int irq, int node, struct io_apic_irq_attr *attr); | 153 | int io_apic_setup_irq_pin_once(unsigned int irq, int node, struct io_apic_irq_attr *attr); |
| 154 | 154 | ||
| 155 | extern struct IO_APIC_route_entry **alloc_ioapic_entries(void); | 155 | extern struct IO_APIC_route_entry **alloc_ioapic_entries(void); |
| 156 | extern void free_ioapic_entries(struct IO_APIC_route_entry **ioapic_entries); | 156 | extern void free_ioapic_entries(struct IO_APIC_route_entry **ioapic_entries); |
diff --git a/arch/x86/kernel/apic/io_apic.c b/arch/x86/kernel/apic/io_apic.c index 68df09bba92e..45fd33d1fd3a 100644 --- a/arch/x86/kernel/apic/io_apic.c +++ b/arch/x86/kernel/apic/io_apic.c | |||
| @@ -128,8 +128,8 @@ static int __init parse_noapic(char *str) | |||
| 128 | } | 128 | } |
| 129 | early_param("noapic", parse_noapic); | 129 | early_param("noapic", parse_noapic); |
| 130 | 130 | ||
| 131 | static int io_apic_setup_irq_pin_once(unsigned int irq, int node, | 131 | static int io_apic_setup_irq_pin(unsigned int irq, int node, |
| 132 | struct io_apic_irq_attr *attr); | 132 | struct io_apic_irq_attr *attr); |
| 133 | 133 | ||
| 134 | /* Will be called in mpparse/acpi/sfi codes for saving IRQ info */ | 134 | /* Will be called in mpparse/acpi/sfi codes for saving IRQ info */ |
| 135 | void mp_save_irq(struct mpc_intsrc *m) | 135 | void mp_save_irq(struct mpc_intsrc *m) |
| @@ -3570,7 +3570,7 @@ int arch_setup_ht_irq(unsigned int irq, struct pci_dev *dev) | |||
| 3570 | } | 3570 | } |
| 3571 | #endif /* CONFIG_HT_IRQ */ | 3571 | #endif /* CONFIG_HT_IRQ */ |
| 3572 | 3572 | ||
| 3573 | int | 3573 | static int |
| 3574 | io_apic_setup_irq_pin(unsigned int irq, int node, struct io_apic_irq_attr *attr) | 3574 | io_apic_setup_irq_pin(unsigned int irq, int node, struct io_apic_irq_attr *attr) |
| 3575 | { | 3575 | { |
| 3576 | struct irq_cfg *cfg = alloc_irq_and_cfg_at(irq, node); | 3576 | struct irq_cfg *cfg = alloc_irq_and_cfg_at(irq, node); |
| @@ -3585,8 +3585,8 @@ io_apic_setup_irq_pin(unsigned int irq, int node, struct io_apic_irq_attr *attr) | |||
| 3585 | return ret; | 3585 | return ret; |
| 3586 | } | 3586 | } |
| 3587 | 3587 | ||
| 3588 | static int io_apic_setup_irq_pin_once(unsigned int irq, int node, | 3588 | int io_apic_setup_irq_pin_once(unsigned int irq, int node, |
| 3589 | struct io_apic_irq_attr *attr) | 3589 | struct io_apic_irq_attr *attr) |
| 3590 | { | 3590 | { |
| 3591 | unsigned int id = attr->ioapic, pin = attr->ioapic_pin; | 3591 | unsigned int id = attr->ioapic, pin = attr->ioapic_pin; |
| 3592 | int ret; | 3592 | int ret; |
diff --git a/arch/x86/kernel/cpu/amd.c b/arch/x86/kernel/cpu/amd.c index 3532d3bf8105..bb9eb29a52dd 100644 --- a/arch/x86/kernel/cpu/amd.c +++ b/arch/x86/kernel/cpu/amd.c | |||
| @@ -698,7 +698,7 @@ cpu_dev_register(amd_cpu_dev); | |||
| 698 | */ | 698 | */ |
| 699 | 699 | ||
| 700 | const int amd_erratum_400[] = | 700 | const int amd_erratum_400[] = |
| 701 | AMD_OSVW_ERRATUM(1, AMD_MODEL_RANGE(0xf, 0x41, 0x2, 0xff, 0xf), | 701 | AMD_OSVW_ERRATUM(1, AMD_MODEL_RANGE(0x0f, 0x4, 0x2, 0xff, 0xf), |
| 702 | AMD_MODEL_RANGE(0x10, 0x2, 0x1, 0xff, 0xf)); | 702 | AMD_MODEL_RANGE(0x10, 0x2, 0x1, 0xff, 0xf)); |
| 703 | EXPORT_SYMBOL_GPL(amd_erratum_400); | 703 | EXPORT_SYMBOL_GPL(amd_erratum_400); |
| 704 | 704 | ||
diff --git a/arch/x86/kernel/cpu/perf_event.c b/arch/x86/kernel/cpu/perf_event.c index 632e5dc9c9c0..e638689279d3 100644 --- a/arch/x86/kernel/cpu/perf_event.c +++ b/arch/x86/kernel/cpu/perf_event.c | |||
| @@ -613,8 +613,8 @@ static int x86_setup_perfctr(struct perf_event *event) | |||
| 613 | /* | 613 | /* |
| 614 | * Branch tracing: | 614 | * Branch tracing: |
| 615 | */ | 615 | */ |
| 616 | if ((attr->config == PERF_COUNT_HW_BRANCH_INSTRUCTIONS) && | 616 | if (attr->config == PERF_COUNT_HW_BRANCH_INSTRUCTIONS && |
| 617 | (hwc->sample_period == 1)) { | 617 | !attr->freq && hwc->sample_period == 1) { |
| 618 | /* BTS is not supported by this architecture. */ | 618 | /* BTS is not supported by this architecture. */ |
| 619 | if (!x86_pmu.bts_active) | 619 | if (!x86_pmu.bts_active) |
| 620 | return -EOPNOTSUPP; | 620 | return -EOPNOTSUPP; |
| @@ -1288,6 +1288,16 @@ static int x86_pmu_handle_irq(struct pt_regs *regs) | |||
| 1288 | 1288 | ||
| 1289 | cpuc = &__get_cpu_var(cpu_hw_events); | 1289 | cpuc = &__get_cpu_var(cpu_hw_events); |
| 1290 | 1290 | ||
| 1291 | /* | ||
| 1292 | * Some chipsets need to unmask the LVTPC in a particular spot | ||
| 1293 | * inside the nmi handler. As a result, the unmasking was pushed | ||
| 1294 | * into all the nmi handlers. | ||
| 1295 | * | ||
| 1296 | * This generic handler doesn't seem to have any issues where the | ||
| 1297 | * unmasking occurs so it was left at the top. | ||
| 1298 | */ | ||
| 1299 | apic_write(APIC_LVTPC, APIC_DM_NMI); | ||
| 1300 | |||
| 1291 | for (idx = 0; idx < x86_pmu.num_counters; idx++) { | 1301 | for (idx = 0; idx < x86_pmu.num_counters; idx++) { |
| 1292 | if (!test_bit(idx, cpuc->active_mask)) { | 1302 | if (!test_bit(idx, cpuc->active_mask)) { |
| 1293 | /* | 1303 | /* |
| @@ -1374,8 +1384,6 @@ perf_event_nmi_handler(struct notifier_block *self, | |||
| 1374 | return NOTIFY_DONE; | 1384 | return NOTIFY_DONE; |
| 1375 | } | 1385 | } |
| 1376 | 1386 | ||
| 1377 | apic_write(APIC_LVTPC, APIC_DM_NMI); | ||
| 1378 | |||
| 1379 | handled = x86_pmu.handle_irq(args->regs); | 1387 | handled = x86_pmu.handle_irq(args->regs); |
| 1380 | if (!handled) | 1388 | if (!handled) |
| 1381 | return NOTIFY_DONE; | 1389 | return NOTIFY_DONE; |
diff --git a/arch/x86/kernel/cpu/perf_event_intel.c b/arch/x86/kernel/cpu/perf_event_intel.c index 43fa20b13817..e61539b07d2c 100644 --- a/arch/x86/kernel/cpu/perf_event_intel.c +++ b/arch/x86/kernel/cpu/perf_event_intel.c | |||
| @@ -25,7 +25,7 @@ struct intel_percore { | |||
| 25 | /* | 25 | /* |
| 26 | * Intel PerfMon, used on Core and later. | 26 | * Intel PerfMon, used on Core and later. |
| 27 | */ | 27 | */ |
| 28 | static const u64 intel_perfmon_event_map[] = | 28 | static u64 intel_perfmon_event_map[PERF_COUNT_HW_MAX] __read_mostly = |
| 29 | { | 29 | { |
| 30 | [PERF_COUNT_HW_CPU_CYCLES] = 0x003c, | 30 | [PERF_COUNT_HW_CPU_CYCLES] = 0x003c, |
| 31 | [PERF_COUNT_HW_INSTRUCTIONS] = 0x00c0, | 31 | [PERF_COUNT_HW_INSTRUCTIONS] = 0x00c0, |
| @@ -933,6 +933,16 @@ static int intel_pmu_handle_irq(struct pt_regs *regs) | |||
| 933 | 933 | ||
| 934 | cpuc = &__get_cpu_var(cpu_hw_events); | 934 | cpuc = &__get_cpu_var(cpu_hw_events); |
| 935 | 935 | ||
| 936 | /* | ||
| 937 | * Some chipsets need to unmask the LVTPC in a particular spot | ||
| 938 | * inside the nmi handler. As a result, the unmasking was pushed | ||
| 939 | * into all the nmi handlers. | ||
| 940 | * | ||
| 941 | * This handler doesn't seem to have any issues with the unmasking | ||
| 942 | * so it was left at the top. | ||
| 943 | */ | ||
| 944 | apic_write(APIC_LVTPC, APIC_DM_NMI); | ||
| 945 | |||
| 936 | intel_pmu_disable_all(); | 946 | intel_pmu_disable_all(); |
| 937 | handled = intel_pmu_drain_bts_buffer(); | 947 | handled = intel_pmu_drain_bts_buffer(); |
| 938 | status = intel_pmu_get_status(); | 948 | status = intel_pmu_get_status(); |
| @@ -998,6 +1008,9 @@ intel_bts_constraints(struct perf_event *event) | |||
| 998 | struct hw_perf_event *hwc = &event->hw; | 1008 | struct hw_perf_event *hwc = &event->hw; |
| 999 | unsigned int hw_event, bts_event; | 1009 | unsigned int hw_event, bts_event; |
| 1000 | 1010 | ||
| 1011 | if (event->attr.freq) | ||
| 1012 | return NULL; | ||
| 1013 | |||
| 1001 | hw_event = hwc->config & INTEL_ARCH_EVENT_MASK; | 1014 | hw_event = hwc->config & INTEL_ARCH_EVENT_MASK; |
| 1002 | bts_event = x86_pmu.event_map(PERF_COUNT_HW_BRANCH_INSTRUCTIONS); | 1015 | bts_event = x86_pmu.event_map(PERF_COUNT_HW_BRANCH_INSTRUCTIONS); |
| 1003 | 1016 | ||
| @@ -1305,7 +1318,7 @@ static void intel_clovertown_quirks(void) | |||
| 1305 | * AJ106 could possibly be worked around by not allowing LBR | 1318 | * AJ106 could possibly be worked around by not allowing LBR |
| 1306 | * usage from PEBS, including the fixup. | 1319 | * usage from PEBS, including the fixup. |
| 1307 | * AJ68 could possibly be worked around by always programming | 1320 | * AJ68 could possibly be worked around by always programming |
| 1308 | * a pebs_event_reset[0] value and coping with the lost events. | 1321 | * a pebs_event_reset[0] value and coping with the lost events. |
| 1309 | * | 1322 | * |
| 1310 | * But taken together it might just make sense to not enable PEBS on | 1323 | * But taken together it might just make sense to not enable PEBS on |
| 1311 | * these chips. | 1324 | * these chips. |
| @@ -1409,6 +1422,18 @@ static __init int intel_pmu_init(void) | |||
| 1409 | x86_pmu.percore_constraints = intel_nehalem_percore_constraints; | 1422 | x86_pmu.percore_constraints = intel_nehalem_percore_constraints; |
| 1410 | x86_pmu.enable_all = intel_pmu_nhm_enable_all; | 1423 | x86_pmu.enable_all = intel_pmu_nhm_enable_all; |
| 1411 | x86_pmu.extra_regs = intel_nehalem_extra_regs; | 1424 | x86_pmu.extra_regs = intel_nehalem_extra_regs; |
| 1425 | |||
| 1426 | if (ebx & 0x40) { | ||
| 1427 | /* | ||
| 1428 | * Erratum AAJ80 detected, we work it around by using | ||
| 1429 | * the BR_MISP_EXEC.ANY event. This will over-count | ||
| 1430 | * branch-misses, but it's still much better than the | ||
| 1431 | * architectural event which is often completely bogus: | ||
| 1432 | */ | ||
| 1433 | intel_perfmon_event_map[PERF_COUNT_HW_BRANCH_MISSES] = 0x7f89; | ||
| 1434 | |||
| 1435 | pr_cont("erratum AAJ80 worked around, "); | ||
| 1436 | } | ||
| 1412 | pr_cont("Nehalem events, "); | 1437 | pr_cont("Nehalem events, "); |
| 1413 | break; | 1438 | break; |
| 1414 | 1439 | ||
diff --git a/arch/x86/kernel/cpu/perf_event_p4.c b/arch/x86/kernel/cpu/perf_event_p4.c index d1f77e2934a1..e93fcd55fae1 100644 --- a/arch/x86/kernel/cpu/perf_event_p4.c +++ b/arch/x86/kernel/cpu/perf_event_p4.c | |||
| @@ -950,11 +950,20 @@ static int p4_pmu_handle_irq(struct pt_regs *regs) | |||
| 950 | x86_pmu_stop(event, 0); | 950 | x86_pmu_stop(event, 0); |
| 951 | } | 951 | } |
| 952 | 952 | ||
| 953 | if (handled) { | 953 | if (handled) |
| 954 | /* p4 quirk: unmask it again */ | ||
| 955 | apic_write(APIC_LVTPC, apic_read(APIC_LVTPC) & ~APIC_LVT_MASKED); | ||
| 956 | inc_irq_stat(apic_perf_irqs); | 954 | inc_irq_stat(apic_perf_irqs); |
| 957 | } | 955 | |
| 956 | /* | ||
| 957 | * When dealing with the unmasking of the LVTPC on P4 perf hw, it has | ||
| 958 | * been observed that the OVF bit flag has to be cleared first _before_ | ||
| 959 | * the LVTPC can be unmasked. | ||
| 960 | * | ||
| 961 | * The reason is the NMI line will continue to be asserted while the OVF | ||
| 962 | * bit is set. This causes a second NMI to generate if the LVTPC is | ||
| 963 | * unmasked before the OVF bit is cleared, leading to unknown NMI | ||
| 964 | * messages. | ||
| 965 | */ | ||
| 966 | apic_write(APIC_LVTPC, APIC_DM_NMI); | ||
| 958 | 967 | ||
| 959 | return handled; | 968 | return handled; |
| 960 | } | 969 | } |
diff --git a/arch/x86/kernel/devicetree.c b/arch/x86/kernel/devicetree.c index 706a9fb46a58..e90f08458e6b 100644 --- a/arch/x86/kernel/devicetree.c +++ b/arch/x86/kernel/devicetree.c | |||
| @@ -391,7 +391,7 @@ static int ioapic_xlate(struct irq_domain *id, const u32 *intspec, u32 intsize, | |||
| 391 | 391 | ||
| 392 | set_io_apic_irq_attr(&attr, idx, line, it->trigger, it->polarity); | 392 | set_io_apic_irq_attr(&attr, idx, line, it->trigger, it->polarity); |
| 393 | 393 | ||
| 394 | return io_apic_setup_irq_pin(*out_hwirq, cpu_to_node(0), &attr); | 394 | return io_apic_setup_irq_pin_once(*out_hwirq, cpu_to_node(0), &attr); |
| 395 | } | 395 | } |
| 396 | 396 | ||
| 397 | static void __init ioapic_add_ofnode(struct device_node *np) | 397 | static void __init ioapic_add_ofnode(struct device_node *np) |
diff --git a/arch/x86/kernel/reboot_32.S b/arch/x86/kernel/reboot_32.S index 29092b38d816..1d5c46df0d78 100644 --- a/arch/x86/kernel/reboot_32.S +++ b/arch/x86/kernel/reboot_32.S | |||
| @@ -21,26 +21,26 @@ r_base = . | |||
| 21 | /* Get our own relocated address */ | 21 | /* Get our own relocated address */ |
| 22 | call 1f | 22 | call 1f |
| 23 | 1: popl %ebx | 23 | 1: popl %ebx |
| 24 | subl $1b, %ebx | 24 | subl $(1b - r_base), %ebx |
| 25 | 25 | ||
| 26 | /* Compute the equivalent real-mode segment */ | 26 | /* Compute the equivalent real-mode segment */ |
| 27 | movl %ebx, %ecx | 27 | movl %ebx, %ecx |
| 28 | shrl $4, %ecx | 28 | shrl $4, %ecx |
| 29 | 29 | ||
| 30 | /* Patch post-real-mode segment jump */ | 30 | /* Patch post-real-mode segment jump */ |
| 31 | movw dispatch_table(%ebx,%eax,2),%ax | 31 | movw (dispatch_table - r_base)(%ebx,%eax,2),%ax |
| 32 | movw %ax, 101f(%ebx) | 32 | movw %ax, (101f - r_base)(%ebx) |
| 33 | movw %cx, 102f(%ebx) | 33 | movw %cx, (102f - r_base)(%ebx) |
| 34 | 34 | ||
| 35 | /* Set up the IDT for real mode. */ | 35 | /* Set up the IDT for real mode. */ |
| 36 | lidtl machine_real_restart_idt(%ebx) | 36 | lidtl (machine_real_restart_idt - r_base)(%ebx) |
| 37 | 37 | ||
| 38 | /* | 38 | /* |
| 39 | * Set up a GDT from which we can load segment descriptors for real | 39 | * Set up a GDT from which we can load segment descriptors for real |
| 40 | * mode. The GDT is not used in real mode; it is just needed here to | 40 | * mode. The GDT is not used in real mode; it is just needed here to |
| 41 | * prepare the descriptors. | 41 | * prepare the descriptors. |
| 42 | */ | 42 | */ |
| 43 | lgdtl machine_real_restart_gdt(%ebx) | 43 | lgdtl (machine_real_restart_gdt - r_base)(%ebx) |
| 44 | 44 | ||
| 45 | /* | 45 | /* |
| 46 | * Load the data segment registers with 16-bit compatible values | 46 | * Load the data segment registers with 16-bit compatible values |
diff --git a/arch/x86/mm/numa_64.c b/arch/x86/mm/numa_64.c index e8c00cc72033..85b52fc03084 100644 --- a/arch/x86/mm/numa_64.c +++ b/arch/x86/mm/numa_64.c | |||
| @@ -306,7 +306,7 @@ int __init numa_cleanup_meminfo(struct numa_meminfo *mi) | |||
| 306 | bi->end = min(bi->end, high); | 306 | bi->end = min(bi->end, high); |
| 307 | 307 | ||
| 308 | /* and there's no empty block */ | 308 | /* and there's no empty block */ |
| 309 | if (bi->start == bi->end) { | 309 | if (bi->start >= bi->end) { |
| 310 | numa_remove_memblk_from(i--, mi); | 310 | numa_remove_memblk_from(i--, mi); |
| 311 | continue; | 311 | continue; |
| 312 | } | 312 | } |
diff --git a/arch/x86/platform/ce4100/falconfalls.dts b/arch/x86/platform/ce4100/falconfalls.dts index 2d6d226f2b10..e70be38ce039 100644 --- a/arch/x86/platform/ce4100/falconfalls.dts +++ b/arch/x86/platform/ce4100/falconfalls.dts | |||
| @@ -347,7 +347,7 @@ | |||
| 347 | "pciclass0c03"; | 347 | "pciclass0c03"; |
| 348 | 348 | ||
| 349 | reg = <0x16800 0x0 0x0 0x0 0x0>; | 349 | reg = <0x16800 0x0 0x0 0x0 0x0>; |
| 350 | interrupts = <22 3>; | 350 | interrupts = <22 1>; |
| 351 | }; | 351 | }; |
| 352 | 352 | ||
| 353 | usb@d,1 { | 353 | usb@d,1 { |
| @@ -357,7 +357,7 @@ | |||
| 357 | "pciclass0c03"; | 357 | "pciclass0c03"; |
| 358 | 358 | ||
| 359 | reg = <0x16900 0x0 0x0 0x0 0x0>; | 359 | reg = <0x16900 0x0 0x0 0x0 0x0>; |
| 360 | interrupts = <22 3>; | 360 | interrupts = <22 1>; |
| 361 | }; | 361 | }; |
| 362 | 362 | ||
| 363 | sata@e,0 { | 363 | sata@e,0 { |
| @@ -367,7 +367,7 @@ | |||
| 367 | "pciclass0106"; | 367 | "pciclass0106"; |
| 368 | 368 | ||
| 369 | reg = <0x17000 0x0 0x0 0x0 0x0>; | 369 | reg = <0x17000 0x0 0x0 0x0 0x0>; |
| 370 | interrupts = <23 3>; | 370 | interrupts = <23 1>; |
| 371 | }; | 371 | }; |
| 372 | 372 | ||
| 373 | flash@f,0 { | 373 | flash@f,0 { |
diff --git a/arch/x86/xen/mmu.c b/arch/x86/xen/mmu.c index aef7af92b28b..55c965b38c27 100644 --- a/arch/x86/xen/mmu.c +++ b/arch/x86/xen/mmu.c | |||
| @@ -1463,6 +1463,119 @@ static int xen_pgd_alloc(struct mm_struct *mm) | |||
| 1463 | return ret; | 1463 | return ret; |
| 1464 | } | 1464 | } |
| 1465 | 1465 | ||
| 1466 | #ifdef CONFIG_X86_64 | ||
| 1467 | static __initdata u64 __last_pgt_set_rw = 0; | ||
| 1468 | static __initdata u64 __pgt_buf_start = 0; | ||
| 1469 | static __initdata u64 __pgt_buf_end = 0; | ||
| 1470 | static __initdata u64 __pgt_buf_top = 0; | ||
| 1471 | /* | ||
| 1472 | * As a consequence of the commit: | ||
| 1473 | * | ||
| 1474 | * commit 4b239f458c229de044d6905c2b0f9fe16ed9e01e | ||
| 1475 | * Author: Yinghai Lu <yinghai@kernel.org> | ||
| 1476 | * Date: Fri Dec 17 16:58:28 2010 -0800 | ||
| 1477 | * | ||
| 1478 | * x86-64, mm: Put early page table high | ||
| 1479 | * | ||
| 1480 | * at some point init_memory_mapping is going to reach the pagetable pages | ||
| 1481 | * area and map those pages too (mapping them as normal memory that falls | ||
| 1482 | * in the range of addresses passed to init_memory_mapping as argument). | ||
| 1483 | * Some of those pages are already pagetable pages (they are in the range | ||
| 1484 | * pgt_buf_start-pgt_buf_end) therefore they are going to be mapped RO and | ||
| 1485 | * everything is fine. | ||
| 1486 | * Some of these pages are not pagetable pages yet (they fall in the range | ||
| 1487 | * pgt_buf_end-pgt_buf_top; for example the page at pgt_buf_end) so they | ||
| 1488 | * are going to be mapped RW. When these pages become pagetable pages and | ||
| 1489 | * are hooked into the pagetable, xen will find that the guest has already | ||
| 1490 | * a RW mapping of them somewhere and fail the operation. | ||
| 1491 | * The reason Xen requires pagetables to be RO is that the hypervisor needs | ||
| 1492 | * to verify that the pagetables are valid before using them. The validation | ||
| 1493 | * operations are called "pinning". | ||
| 1494 | * | ||
| 1495 | * In order to fix the issue we mark all the pages in the entire range | ||
| 1496 | * pgt_buf_start-pgt_buf_top as RO, however when the pagetable allocation | ||
| 1497 | * is completed only the range pgt_buf_start-pgt_buf_end is reserved by | ||
| 1498 | * init_memory_mapping. Hence the kernel is going to crash as soon as one | ||
| 1499 | * of the pages in the range pgt_buf_end-pgt_buf_top is reused (b/c those | ||
| 1500 | * ranges are RO). | ||
| 1501 | * | ||
| 1502 | * For this reason, 'mark_rw_past_pgt' is introduced which is called _after_ | ||
| 1503 | * the init_memory_mapping has completed (in a perfect world we would | ||
| 1504 | * call this function from init_memory_mapping, but lets ignore that). | ||
| 1505 | * | ||
| 1506 | * Because we are called _after_ init_memory_mapping the pgt_buf_[start, | ||
| 1507 | * end,top] have all changed to new values (b/c init_memory_mapping | ||
| 1508 | * is called and setting up another new page-table). Hence, the first time | ||
| 1509 | * we enter this function, we save away the pgt_buf_start value and update | ||
| 1510 | * the pgt_buf_[end,top]. | ||
| 1511 | * | ||
| 1512 | * When we detect that the "old" pgt_buf_start through pgt_buf_end | ||
| 1513 | * PFNs have been reserved (so memblock_x86_reserve_range has been called), | ||
| 1514 | * we immediately set out to RW the "old" pgt_buf_end through pgt_buf_top. | ||
| 1515 | * | ||
| 1516 | * And then we update those "old" pgt_buf_[end|top] with the new ones | ||
| 1517 | * so that we can redo this on the next pagetable. | ||
| 1518 | */ | ||
| 1519 | static __init void mark_rw_past_pgt(void) { | ||
| 1520 | |||
| 1521 | if (pgt_buf_end > pgt_buf_start) { | ||
| 1522 | u64 addr, size; | ||
| 1523 | |||
| 1524 | /* Save it away. */ | ||
| 1525 | if (!__pgt_buf_start) { | ||
| 1526 | __pgt_buf_start = pgt_buf_start; | ||
| 1527 | __pgt_buf_end = pgt_buf_end; | ||
| 1528 | __pgt_buf_top = pgt_buf_top; | ||
| 1529 | return; | ||
| 1530 | } | ||
| 1531 | /* If we get the range that starts at __pgt_buf_end that means | ||
| 1532 | * the range is reserved, and that in 'init_memory_mapping' | ||
| 1533 | * the 'memblock_x86_reserve_range' has been called with the | ||
| 1534 | * outdated __pgt_buf_start, __pgt_buf_end (the "new" | ||
| 1535 | * pgt_buf_[start|end|top] refer now to a new pagetable. | ||
| 1536 | * Note: we are called _after_ the pgt_buf_[..] have been | ||
| 1537 | * updated.*/ | ||
| 1538 | |||
| 1539 | addr = memblock_x86_find_in_range_size(PFN_PHYS(__pgt_buf_start), | ||
| 1540 | &size, PAGE_SIZE); | ||
| 1541 | |||
| 1542 | /* Still not reserved, meaning 'memblock_x86_reserve_range' | ||
| 1543 | * hasn't been called yet. Update the _end and _top.*/ | ||
| 1544 | if (addr == PFN_PHYS(__pgt_buf_start)) { | ||
| 1545 | __pgt_buf_end = pgt_buf_end; | ||
| 1546 | __pgt_buf_top = pgt_buf_top; | ||
| 1547 | return; | ||
| 1548 | } | ||
| 1549 | |||
| 1550 | /* OK, the area is reserved, meaning it is time for us to | ||
| 1551 | * set RW for the old end->top PFNs. */ | ||
| 1552 | |||
| 1553 | /* ..unless we had already done this. */ | ||
| 1554 | if (__pgt_buf_end == __last_pgt_set_rw) | ||
| 1555 | return; | ||
| 1556 | |||
| 1557 | addr = PFN_PHYS(__pgt_buf_end); | ||
| 1558 | |||
| 1559 | /* set as RW the rest */ | ||
| 1560 | printk(KERN_DEBUG "xen: setting RW the range %llx - %llx\n", | ||
| 1561 | PFN_PHYS(__pgt_buf_end), PFN_PHYS(__pgt_buf_top)); | ||
| 1562 | |||
| 1563 | while (addr < PFN_PHYS(__pgt_buf_top)) { | ||
| 1564 | make_lowmem_page_readwrite(__va(addr)); | ||
| 1565 | addr += PAGE_SIZE; | ||
| 1566 | } | ||
| 1567 | /* And update everything so that we are ready for the next | ||
| 1568 | * pagetable (the one created for regions past 4GB) */ | ||
| 1569 | __last_pgt_set_rw = __pgt_buf_end; | ||
| 1570 | __pgt_buf_start = pgt_buf_start; | ||
| 1571 | __pgt_buf_end = pgt_buf_end; | ||
| 1572 | __pgt_buf_top = pgt_buf_top; | ||
| 1573 | } | ||
| 1574 | return; | ||
| 1575 | } | ||
| 1576 | #else | ||
| 1577 | static __init void mark_rw_past_pgt(void) { } | ||
| 1578 | #endif | ||
| 1466 | static void xen_pgd_free(struct mm_struct *mm, pgd_t *pgd) | 1579 | static void xen_pgd_free(struct mm_struct *mm, pgd_t *pgd) |
| 1467 | { | 1580 | { |
| 1468 | #ifdef CONFIG_X86_64 | 1581 | #ifdef CONFIG_X86_64 |
| @@ -1489,13 +1602,21 @@ static __init pte_t mask_rw_pte(pte_t *ptep, pte_t pte) | |||
| 1489 | unsigned long pfn = pte_pfn(pte); | 1602 | unsigned long pfn = pte_pfn(pte); |
| 1490 | 1603 | ||
| 1491 | /* | 1604 | /* |
| 1605 | * A bit of optimization. We do not need to call the workaround | ||
| 1606 | * when xen_set_pte_init is called with a PTE with 0 as PFN. | ||
| 1607 | * That is b/c the pagetable at that point are just being populated | ||
| 1608 | * with empty values and we can save some cycles by not calling | ||
| 1609 | * the 'memblock' code.*/ | ||
| 1610 | if (pfn) | ||
| 1611 | mark_rw_past_pgt(); | ||
| 1612 | /* | ||
| 1492 | * If the new pfn is within the range of the newly allocated | 1613 | * If the new pfn is within the range of the newly allocated |
| 1493 | * kernel pagetable, and it isn't being mapped into an | 1614 | * kernel pagetable, and it isn't being mapped into an |
| 1494 | * early_ioremap fixmap slot as a freshly allocated page, make sure | 1615 | * early_ioremap fixmap slot as a freshly allocated page, make sure |
| 1495 | * it is RO. | 1616 | * it is RO. |
| 1496 | */ | 1617 | */ |
| 1497 | if (((!is_early_ioremap_ptep(ptep) && | 1618 | if (((!is_early_ioremap_ptep(ptep) && |
| 1498 | pfn >= pgt_buf_start && pfn < pgt_buf_end)) || | 1619 | pfn >= pgt_buf_start && pfn < pgt_buf_top)) || |
| 1499 | (is_early_ioremap_ptep(ptep) && pfn != (pgt_buf_end - 1))) | 1620 | (is_early_ioremap_ptep(ptep) && pfn != (pgt_buf_end - 1))) |
| 1500 | pte = pte_wrprotect(pte); | 1621 | pte = pte_wrprotect(pte); |
| 1501 | 1622 | ||
| @@ -1997,6 +2118,8 @@ __init void xen_ident_map_ISA(void) | |||
| 1997 | 2118 | ||
| 1998 | static __init void xen_post_allocator_init(void) | 2119 | static __init void xen_post_allocator_init(void) |
| 1999 | { | 2120 | { |
| 2121 | mark_rw_past_pgt(); | ||
| 2122 | |||
| 2000 | #ifdef CONFIG_XEN_DEBUG | 2123 | #ifdef CONFIG_XEN_DEBUG |
| 2001 | pv_mmu_ops.make_pte = PV_CALLEE_SAVE(xen_make_pte_debug); | 2124 | pv_mmu_ops.make_pte = PV_CALLEE_SAVE(xen_make_pte_debug); |
| 2002 | #endif | 2125 | #endif |
diff --git a/drivers/acpi/scan.c b/drivers/acpi/scan.c index b136c9c1e531..449c556274c0 100644 --- a/drivers/acpi/scan.c +++ b/drivers/acpi/scan.c | |||
| @@ -943,6 +943,10 @@ static int acpi_bus_get_flags(struct acpi_device *device) | |||
| 943 | if (ACPI_SUCCESS(status)) | 943 | if (ACPI_SUCCESS(status)) |
| 944 | device->flags.lockable = 1; | 944 | device->flags.lockable = 1; |
| 945 | 945 | ||
| 946 | /* Power resources cannot be power manageable. */ | ||
| 947 | if (device->device_type == ACPI_BUS_TYPE_POWER) | ||
| 948 | return 0; | ||
| 949 | |||
| 946 | /* Presence of _PS0|_PR0 indicates 'power manageable' */ | 950 | /* Presence of _PS0|_PR0 indicates 'power manageable' */ |
| 947 | status = acpi_get_handle(device->handle, "_PS0", &temp); | 951 | status = acpi_get_handle(device->handle, "_PS0", &temp); |
| 948 | if (ACPI_FAILURE(status)) | 952 | if (ACPI_FAILURE(status)) |
diff --git a/drivers/base/power/main.c b/drivers/base/power/main.c index fbc5b6e7c591..abe3ab709e87 100644 --- a/drivers/base/power/main.c +++ b/drivers/base/power/main.c | |||
| @@ -63,6 +63,7 @@ void device_pm_init(struct device *dev) | |||
| 63 | dev->power.wakeup = NULL; | 63 | dev->power.wakeup = NULL; |
| 64 | spin_lock_init(&dev->power.lock); | 64 | spin_lock_init(&dev->power.lock); |
| 65 | pm_runtime_init(dev); | 65 | pm_runtime_init(dev); |
| 66 | INIT_LIST_HEAD(&dev->power.entry); | ||
| 66 | } | 67 | } |
| 67 | 68 | ||
| 68 | /** | 69 | /** |
diff --git a/drivers/base/power/wakeup.c b/drivers/base/power/wakeup.c index 4573c83df6dd..abbbd33e8d8a 100644 --- a/drivers/base/power/wakeup.c +++ b/drivers/base/power/wakeup.c | |||
| @@ -258,7 +258,7 @@ void device_set_wakeup_capable(struct device *dev, bool capable) | |||
| 258 | if (!!dev->power.can_wakeup == !!capable) | 258 | if (!!dev->power.can_wakeup == !!capable) |
| 259 | return; | 259 | return; |
| 260 | 260 | ||
| 261 | if (device_is_registered(dev)) { | 261 | if (device_is_registered(dev) && !list_empty(&dev->power.entry)) { |
| 262 | if (capable) { | 262 | if (capable) { |
| 263 | if (wakeup_sysfs_add(dev)) | 263 | if (wakeup_sysfs_add(dev)) |
| 264 | return; | 264 | return; |
diff --git a/drivers/char/agp/intel-agp.c b/drivers/char/agp/intel-agp.c index b0a0dccc98c1..b427711be4be 100644 --- a/drivers/char/agp/intel-agp.c +++ b/drivers/char/agp/intel-agp.c | |||
| @@ -903,6 +903,9 @@ static struct pci_device_id agp_intel_pci_table[] = { | |||
| 903 | ID(PCI_DEVICE_ID_INTEL_SANDYBRIDGE_HB), | 903 | ID(PCI_DEVICE_ID_INTEL_SANDYBRIDGE_HB), |
| 904 | ID(PCI_DEVICE_ID_INTEL_SANDYBRIDGE_M_HB), | 904 | ID(PCI_DEVICE_ID_INTEL_SANDYBRIDGE_M_HB), |
| 905 | ID(PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_HB), | 905 | ID(PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_HB), |
| 906 | ID(PCI_DEVICE_ID_INTEL_IVYBRIDGE_HB), | ||
| 907 | ID(PCI_DEVICE_ID_INTEL_IVYBRIDGE_M_HB), | ||
| 908 | ID(PCI_DEVICE_ID_INTEL_IVYBRIDGE_S_HB), | ||
| 906 | { } | 909 | { } |
| 907 | }; | 910 | }; |
| 908 | 911 | ||
diff --git a/drivers/char/agp/intel-agp.h b/drivers/char/agp/intel-agp.h index 5feebe2800e9..999803ce10dc 100644 --- a/drivers/char/agp/intel-agp.h +++ b/drivers/char/agp/intel-agp.h | |||
| @@ -225,6 +225,14 @@ | |||
| 225 | #define PCI_DEVICE_ID_INTEL_SANDYBRIDGE_M_GT2_PLUS_IG 0x0126 | 225 | #define PCI_DEVICE_ID_INTEL_SANDYBRIDGE_M_GT2_PLUS_IG 0x0126 |
| 226 | #define PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_HB 0x0108 /* Server */ | 226 | #define PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_HB 0x0108 /* Server */ |
| 227 | #define PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_IG 0x010A | 227 | #define PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_IG 0x010A |
| 228 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_HB 0x0150 /* Desktop */ | ||
| 229 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_GT1_IG 0x0152 | ||
| 230 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_GT2_IG 0x0162 | ||
| 231 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_M_HB 0x0154 /* Mobile */ | ||
| 232 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_M_GT1_IG 0x0156 | ||
| 233 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_M_GT2_IG 0x0166 | ||
| 234 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_S_HB 0x0158 /* Server */ | ||
| 235 | #define PCI_DEVICE_ID_INTEL_IVYBRIDGE_S_GT1_IG 0x015A | ||
| 228 | 236 | ||
| 229 | int intel_gmch_probe(struct pci_dev *pdev, | 237 | int intel_gmch_probe(struct pci_dev *pdev, |
| 230 | struct agp_bridge_data *bridge); | 238 | struct agp_bridge_data *bridge); |
diff --git a/drivers/char/agp/intel-gtt.c b/drivers/char/agp/intel-gtt.c index 0d09b537bb9a..85151019dde1 100644 --- a/drivers/char/agp/intel-gtt.c +++ b/drivers/char/agp/intel-gtt.c | |||
| @@ -1420,6 +1420,16 @@ static const struct intel_gtt_driver_description { | |||
| 1420 | "Sandybridge", &sandybridge_gtt_driver }, | 1420 | "Sandybridge", &sandybridge_gtt_driver }, |
| 1421 | { PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_IG, | 1421 | { PCI_DEVICE_ID_INTEL_SANDYBRIDGE_S_IG, |
| 1422 | "Sandybridge", &sandybridge_gtt_driver }, | 1422 | "Sandybridge", &sandybridge_gtt_driver }, |
| 1423 | { PCI_DEVICE_ID_INTEL_IVYBRIDGE_GT1_IG, | ||
| 1424 | "Ivybridge", &sandybridge_gtt_driver }, | ||
| 1425 | { PCI_DEVICE_ID_INTEL_IVYBRIDGE_GT2_IG, | ||
| 1426 | "Ivybridge", &sandybridge_gtt_driver }, | ||
| 1427 | { PCI_DEVICE_ID_INTEL_IVYBRIDGE_M_GT1_IG, | ||
| 1428 | "Ivybridge", &sandybridge_gtt_driver }, | ||
| 1429 | { PCI_DEVICE_ID_INTEL_IVYBRIDGE_M_GT2_IG, | ||
| 1430 | "Ivybridge", &sandybridge_gtt_driver }, | ||
| 1431 | { PCI_DEVICE_ID_INTEL_IVYBRIDGE_S_GT1_IG, | ||
| 1432 | "Ivybridge", &sandybridge_gtt_driver }, | ||
| 1423 | { 0, NULL, NULL } | 1433 | { 0, NULL, NULL } |
| 1424 | }; | 1434 | }; |
| 1425 | 1435 | ||
diff --git a/drivers/clk/clkdev.c b/drivers/clk/clkdev.c index 0fc0a79852de..6db161f64ae0 100644 --- a/drivers/clk/clkdev.c +++ b/drivers/clk/clkdev.c | |||
| @@ -32,10 +32,9 @@ static DEFINE_MUTEX(clocks_mutex); | |||
| 32 | * Then we take the most specific entry - with the following | 32 | * Then we take the most specific entry - with the following |
| 33 | * order of precedence: dev+con > dev only > con only. | 33 | * order of precedence: dev+con > dev only > con only. |
| 34 | */ | 34 | */ |
| 35 | static struct clk *clk_find(const char *dev_id, const char *con_id) | 35 | static struct clk_lookup *clk_find(const char *dev_id, const char *con_id) |
| 36 | { | 36 | { |
| 37 | struct clk_lookup *p; | 37 | struct clk_lookup *p, *cl = NULL; |
| 38 | struct clk *clk = NULL; | ||
| 39 | int match, best = 0; | 38 | int match, best = 0; |
| 40 | 39 | ||
| 41 | list_for_each_entry(p, &clocks, node) { | 40 | list_for_each_entry(p, &clocks, node) { |
| @@ -52,27 +51,27 @@ static struct clk *clk_find(const char *dev_id, const char *con_id) | |||
| 52 | } | 51 | } |
| 53 | 52 | ||
| 54 | if (match > best) { | 53 | if (match > best) { |
| 55 | clk = p->clk; | 54 | cl = p; |
| 56 | if (match != 3) | 55 | if (match != 3) |
| 57 | best = match; | 56 | best = match; |
| 58 | else | 57 | else |
| 59 | break; | 58 | break; |
| 60 | } | 59 | } |
| 61 | } | 60 | } |
| 62 | return clk; | 61 | return cl; |
| 63 | } | 62 | } |
| 64 | 63 | ||
| 65 | struct clk *clk_get_sys(const char *dev_id, const char *con_id) | 64 | struct clk *clk_get_sys(const char *dev_id, const char *con_id) |
| 66 | { | 65 | { |
| 67 | struct clk *clk; | 66 | struct clk_lookup *cl; |
| 68 | 67 | ||
| 69 | mutex_lock(&clocks_mutex); | 68 | mutex_lock(&clocks_mutex); |
| 70 | clk = clk_find(dev_id, con_id); | 69 | cl = clk_find(dev_id, con_id); |
| 71 | if (clk && !__clk_get(clk)) | 70 | if (cl && !__clk_get(cl->clk)) |
| 72 | clk = NULL; | 71 | cl = NULL; |
| 73 | mutex_unlock(&clocks_mutex); | 72 | mutex_unlock(&clocks_mutex); |
| 74 | 73 | ||
| 75 | return clk ? clk : ERR_PTR(-ENOENT); | 74 | return cl ? cl->clk : ERR_PTR(-ENOENT); |
| 76 | } | 75 | } |
| 77 | EXPORT_SYMBOL(clk_get_sys); | 76 | EXPORT_SYMBOL(clk_get_sys); |
| 78 | 77 | ||
diff --git a/drivers/gpu/drm/drm_irq.c b/drivers/gpu/drm/drm_irq.c index 62ced7554ba4..2022a5c966bb 100644 --- a/drivers/gpu/drm/drm_irq.c +++ b/drivers/gpu/drm/drm_irq.c | |||
| @@ -933,11 +933,34 @@ EXPORT_SYMBOL(drm_vblank_put); | |||
| 933 | 933 | ||
| 934 | void drm_vblank_off(struct drm_device *dev, int crtc) | 934 | void drm_vblank_off(struct drm_device *dev, int crtc) |
| 935 | { | 935 | { |
| 936 | struct drm_pending_vblank_event *e, *t; | ||
| 937 | struct timeval now; | ||
| 936 | unsigned long irqflags; | 938 | unsigned long irqflags; |
| 939 | unsigned int seq; | ||
| 937 | 940 | ||
| 938 | spin_lock_irqsave(&dev->vbl_lock, irqflags); | 941 | spin_lock_irqsave(&dev->vbl_lock, irqflags); |
| 939 | vblank_disable_and_save(dev, crtc); | 942 | vblank_disable_and_save(dev, crtc); |
| 940 | DRM_WAKEUP(&dev->vbl_queue[crtc]); | 943 | DRM_WAKEUP(&dev->vbl_queue[crtc]); |
| 944 | |||
| 945 | /* Send any queued vblank events, lest the natives grow disquiet */ | ||
| 946 | seq = drm_vblank_count_and_time(dev, crtc, &now); | ||
| 947 | list_for_each_entry_safe(e, t, &dev->vblank_event_list, base.link) { | ||
| 948 | if (e->pipe != crtc) | ||
| 949 | continue; | ||
| 950 | DRM_DEBUG("Sending premature vblank event on disable: \ | ||
| 951 | wanted %d, current %d\n", | ||
| 952 | e->event.sequence, seq); | ||
| 953 | |||
| 954 | e->event.sequence = seq; | ||
| 955 | e->event.tv_sec = now.tv_sec; | ||
| 956 | e->event.tv_usec = now.tv_usec; | ||
| 957 | drm_vblank_put(dev, e->pipe); | ||
| 958 | list_move_tail(&e->base.link, &e->base.file_priv->event_list); | ||
| 959 | wake_up_interruptible(&e->base.file_priv->event_wait); | ||
| 960 | trace_drm_vblank_event_delivered(e->base.pid, e->pipe, | ||
| 961 | e->event.sequence); | ||
| 962 | } | ||
| 963 | |||
| 941 | spin_unlock_irqrestore(&dev->vbl_lock, irqflags); | 964 | spin_unlock_irqrestore(&dev->vbl_lock, irqflags); |
| 942 | } | 965 | } |
| 943 | EXPORT_SYMBOL(drm_vblank_off); | 966 | EXPORT_SYMBOL(drm_vblank_off); |
diff --git a/drivers/gpu/drm/i915/i915_debugfs.c b/drivers/gpu/drm/i915/i915_debugfs.c index 87c8e29465e3..183eaac8980a 100644 --- a/drivers/gpu/drm/i915/i915_debugfs.c +++ b/drivers/gpu/drm/i915/i915_debugfs.c | |||
| @@ -106,11 +106,12 @@ static const char *get_tiling_flag(struct drm_i915_gem_object *obj) | |||
| 106 | } | 106 | } |
| 107 | } | 107 | } |
| 108 | 108 | ||
| 109 | static const char *agp_type_str(int type) | 109 | static const char *cache_level_str(int type) |
| 110 | { | 110 | { |
| 111 | switch (type) { | 111 | switch (type) { |
| 112 | case 0: return " uncached"; | 112 | case I915_CACHE_NONE: return " uncached"; |
| 113 | case 1: return " snooped"; | 113 | case I915_CACHE_LLC: return " snooped (LLC)"; |
| 114 | case I915_CACHE_LLC_MLC: return " snooped (LLC+MLC)"; | ||
| 114 | default: return ""; | 115 | default: return ""; |
| 115 | } | 116 | } |
| 116 | } | 117 | } |
| @@ -127,7 +128,7 @@ describe_obj(struct seq_file *m, struct drm_i915_gem_object *obj) | |||
| 127 | obj->base.write_domain, | 128 | obj->base.write_domain, |
| 128 | obj->last_rendering_seqno, | 129 | obj->last_rendering_seqno, |
| 129 | obj->last_fenced_seqno, | 130 | obj->last_fenced_seqno, |
| 130 | agp_type_str(obj->agp_type == AGP_USER_CACHED_MEMORY), | 131 | cache_level_str(obj->cache_level), |
| 131 | obj->dirty ? " dirty" : "", | 132 | obj->dirty ? " dirty" : "", |
| 132 | obj->madv == I915_MADV_DONTNEED ? " purgeable" : ""); | 133 | obj->madv == I915_MADV_DONTNEED ? " purgeable" : ""); |
| 133 | if (obj->base.name) | 134 | if (obj->base.name) |
| @@ -714,7 +715,7 @@ static void print_error_buffers(struct seq_file *m, | |||
| 714 | dirty_flag(err->dirty), | 715 | dirty_flag(err->dirty), |
| 715 | purgeable_flag(err->purgeable), | 716 | purgeable_flag(err->purgeable), |
| 716 | ring_str(err->ring), | 717 | ring_str(err->ring), |
| 717 | agp_type_str(err->agp_type)); | 718 | cache_level_str(err->cache_level)); |
| 718 | 719 | ||
| 719 | if (err->name) | 720 | if (err->name) |
| 720 | seq_printf(m, " (name: %d)", err->name); | 721 | seq_printf(m, " (name: %d)", err->name); |
| @@ -852,6 +853,7 @@ static int i915_cur_delayinfo(struct seq_file *m, void *unused) | |||
| 852 | struct drm_info_node *node = (struct drm_info_node *) m->private; | 853 | struct drm_info_node *node = (struct drm_info_node *) m->private; |
| 853 | struct drm_device *dev = node->minor->dev; | 854 | struct drm_device *dev = node->minor->dev; |
| 854 | drm_i915_private_t *dev_priv = dev->dev_private; | 855 | drm_i915_private_t *dev_priv = dev->dev_private; |
| 856 | int ret; | ||
| 855 | 857 | ||
| 856 | if (IS_GEN5(dev)) { | 858 | if (IS_GEN5(dev)) { |
| 857 | u16 rgvswctl = I915_READ16(MEMSWCTL); | 859 | u16 rgvswctl = I915_READ16(MEMSWCTL); |
| @@ -873,7 +875,11 @@ static int i915_cur_delayinfo(struct seq_file *m, void *unused) | |||
| 873 | int max_freq; | 875 | int max_freq; |
| 874 | 876 | ||
| 875 | /* RPSTAT1 is in the GT power well */ | 877 | /* RPSTAT1 is in the GT power well */ |
| 876 | __gen6_gt_force_wake_get(dev_priv); | 878 | ret = mutex_lock_interruptible(&dev->struct_mutex); |
| 879 | if (ret) | ||
| 880 | return ret; | ||
| 881 | |||
| 882 | gen6_gt_force_wake_get(dev_priv); | ||
| 877 | 883 | ||
| 878 | rpstat = I915_READ(GEN6_RPSTAT1); | 884 | rpstat = I915_READ(GEN6_RPSTAT1); |
| 879 | rpupei = I915_READ(GEN6_RP_CUR_UP_EI); | 885 | rpupei = I915_READ(GEN6_RP_CUR_UP_EI); |
| @@ -883,6 +889,9 @@ static int i915_cur_delayinfo(struct seq_file *m, void *unused) | |||
| 883 | rpcurdown = I915_READ(GEN6_RP_CUR_DOWN); | 889 | rpcurdown = I915_READ(GEN6_RP_CUR_DOWN); |
| 884 | rpprevdown = I915_READ(GEN6_RP_PREV_DOWN); | 890 | rpprevdown = I915_READ(GEN6_RP_PREV_DOWN); |
| 885 | 891 | ||
| 892 | gen6_gt_force_wake_put(dev_priv); | ||
| 893 | mutex_unlock(&dev->struct_mutex); | ||
| 894 | |||
| 886 | seq_printf(m, "GT_PERF_STATUS: 0x%08x\n", gt_perf_status); | 895 | seq_printf(m, "GT_PERF_STATUS: 0x%08x\n", gt_perf_status); |
| 887 | seq_printf(m, "RPSTAT1: 0x%08x\n", rpstat); | 896 | seq_printf(m, "RPSTAT1: 0x%08x\n", rpstat); |
| 888 | seq_printf(m, "Render p-state ratio: %d\n", | 897 | seq_printf(m, "Render p-state ratio: %d\n", |
| @@ -917,8 +926,6 @@ static int i915_cur_delayinfo(struct seq_file *m, void *unused) | |||
| 917 | max_freq = rp_state_cap & 0xff; | 926 | max_freq = rp_state_cap & 0xff; |
| 918 | seq_printf(m, "Max non-overclocked (RP0) frequency: %dMHz\n", | 927 | seq_printf(m, "Max non-overclocked (RP0) frequency: %dMHz\n", |
| 919 | max_freq * 50); | 928 | max_freq * 50); |
| 920 | |||
| 921 | __gen6_gt_force_wake_put(dev_priv); | ||
| 922 | } else { | 929 | } else { |
| 923 | seq_printf(m, "no P-state info available\n"); | 930 | seq_printf(m, "no P-state info available\n"); |
| 924 | } | 931 | } |
| @@ -1186,6 +1193,42 @@ static int i915_gem_framebuffer_info(struct seq_file *m, void *data) | |||
| 1186 | return 0; | 1193 | return 0; |
| 1187 | } | 1194 | } |
| 1188 | 1195 | ||
| 1196 | static int i915_context_status(struct seq_file *m, void *unused) | ||
| 1197 | { | ||
| 1198 | struct drm_info_node *node = (struct drm_info_node *) m->private; | ||
| 1199 | struct drm_device *dev = node->minor->dev; | ||
| 1200 | drm_i915_private_t *dev_priv = dev->dev_private; | ||
| 1201 | int ret; | ||
| 1202 | |||
| 1203 | ret = mutex_lock_interruptible(&dev->mode_config.mutex); | ||
| 1204 | if (ret) | ||
| 1205 | return ret; | ||
| 1206 | |||
| 1207 | seq_printf(m, "power context "); | ||
| 1208 | describe_obj(m, dev_priv->pwrctx); | ||
| 1209 | seq_printf(m, "\n"); | ||
| 1210 | |||
| 1211 | seq_printf(m, "render context "); | ||
| 1212 | describe_obj(m, dev_priv->renderctx); | ||
| 1213 | seq_printf(m, "\n"); | ||
| 1214 | |||
| 1215 | mutex_unlock(&dev->mode_config.mutex); | ||
| 1216 | |||
| 1217 | return 0; | ||
| 1218 | } | ||
| 1219 | |||
| 1220 | static int i915_gen6_forcewake_count_info(struct seq_file *m, void *data) | ||
| 1221 | { | ||
| 1222 | struct drm_info_node *node = (struct drm_info_node *) m->private; | ||
| 1223 | struct drm_device *dev = node->minor->dev; | ||
| 1224 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 1225 | |||
| 1226 | seq_printf(m, "forcewake count = %d\n", | ||
| 1227 | atomic_read(&dev_priv->forcewake_count)); | ||
| 1228 | |||
| 1229 | return 0; | ||
| 1230 | } | ||
| 1231 | |||
| 1189 | static int | 1232 | static int |
| 1190 | i915_wedged_open(struct inode *inode, | 1233 | i915_wedged_open(struct inode *inode, |
| 1191 | struct file *filp) | 1234 | struct file *filp) |
| @@ -1288,6 +1331,67 @@ static int i915_wedged_create(struct dentry *root, struct drm_minor *minor) | |||
| 1288 | return drm_add_fake_info_node(minor, ent, &i915_wedged_fops); | 1331 | return drm_add_fake_info_node(minor, ent, &i915_wedged_fops); |
| 1289 | } | 1332 | } |
| 1290 | 1333 | ||
| 1334 | static int i915_forcewake_open(struct inode *inode, struct file *file) | ||
| 1335 | { | ||
| 1336 | struct drm_device *dev = inode->i_private; | ||
| 1337 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 1338 | int ret; | ||
| 1339 | |||
| 1340 | if (!IS_GEN6(dev)) | ||
| 1341 | return 0; | ||
| 1342 | |||
| 1343 | ret = mutex_lock_interruptible(&dev->struct_mutex); | ||
| 1344 | if (ret) | ||
| 1345 | return ret; | ||
| 1346 | gen6_gt_force_wake_get(dev_priv); | ||
| 1347 | mutex_unlock(&dev->struct_mutex); | ||
| 1348 | |||
| 1349 | return 0; | ||
| 1350 | } | ||
| 1351 | |||
| 1352 | int i915_forcewake_release(struct inode *inode, struct file *file) | ||
| 1353 | { | ||
| 1354 | struct drm_device *dev = inode->i_private; | ||
| 1355 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 1356 | |||
| 1357 | if (!IS_GEN6(dev)) | ||
| 1358 | return 0; | ||
| 1359 | |||
| 1360 | /* | ||
| 1361 | * It's bad that we can potentially hang userspace if struct_mutex gets | ||
| 1362 | * forever stuck. However, if we cannot acquire this lock it means that | ||
| 1363 | * almost certainly the driver has hung, is not unload-able. Therefore | ||
| 1364 | * hanging here is probably a minor inconvenience not to be seen my | ||
| 1365 | * almost every user. | ||
| 1366 | */ | ||
| 1367 | mutex_lock(&dev->struct_mutex); | ||
| 1368 | gen6_gt_force_wake_put(dev_priv); | ||
| 1369 | mutex_unlock(&dev->struct_mutex); | ||
| 1370 | |||
| 1371 | return 0; | ||
| 1372 | } | ||
| 1373 | |||
| 1374 | static const struct file_operations i915_forcewake_fops = { | ||
| 1375 | .owner = THIS_MODULE, | ||
| 1376 | .open = i915_forcewake_open, | ||
| 1377 | .release = i915_forcewake_release, | ||
| 1378 | }; | ||
| 1379 | |||
| 1380 | static int i915_forcewake_create(struct dentry *root, struct drm_minor *minor) | ||
| 1381 | { | ||
| 1382 | struct drm_device *dev = minor->dev; | ||
| 1383 | struct dentry *ent; | ||
| 1384 | |||
| 1385 | ent = debugfs_create_file("i915_forcewake_user", | ||
| 1386 | S_IRUSR, | ||
| 1387 | root, dev, | ||
| 1388 | &i915_forcewake_fops); | ||
| 1389 | if (IS_ERR(ent)) | ||
| 1390 | return PTR_ERR(ent); | ||
| 1391 | |||
| 1392 | return drm_add_fake_info_node(minor, ent, &i915_forcewake_fops); | ||
| 1393 | } | ||
| 1394 | |||
| 1291 | static struct drm_info_list i915_debugfs_list[] = { | 1395 | static struct drm_info_list i915_debugfs_list[] = { |
| 1292 | {"i915_capabilities", i915_capabilities, 0}, | 1396 | {"i915_capabilities", i915_capabilities, 0}, |
| 1293 | {"i915_gem_objects", i915_gem_object_info, 0}, | 1397 | {"i915_gem_objects", i915_gem_object_info, 0}, |
| @@ -1324,6 +1428,8 @@ static struct drm_info_list i915_debugfs_list[] = { | |||
| 1324 | {"i915_sr_status", i915_sr_status, 0}, | 1428 | {"i915_sr_status", i915_sr_status, 0}, |
| 1325 | {"i915_opregion", i915_opregion, 0}, | 1429 | {"i915_opregion", i915_opregion, 0}, |
| 1326 | {"i915_gem_framebuffer", i915_gem_framebuffer_info, 0}, | 1430 | {"i915_gem_framebuffer", i915_gem_framebuffer_info, 0}, |
| 1431 | {"i915_context_status", i915_context_status, 0}, | ||
| 1432 | {"i915_gen6_forcewake_count", i915_gen6_forcewake_count_info, 0}, | ||
| 1327 | }; | 1433 | }; |
| 1328 | #define I915_DEBUGFS_ENTRIES ARRAY_SIZE(i915_debugfs_list) | 1434 | #define I915_DEBUGFS_ENTRIES ARRAY_SIZE(i915_debugfs_list) |
| 1329 | 1435 | ||
| @@ -1335,6 +1441,10 @@ int i915_debugfs_init(struct drm_minor *minor) | |||
| 1335 | if (ret) | 1441 | if (ret) |
| 1336 | return ret; | 1442 | return ret; |
| 1337 | 1443 | ||
| 1444 | ret = i915_forcewake_create(minor->debugfs_root, minor); | ||
| 1445 | if (ret) | ||
| 1446 | return ret; | ||
| 1447 | |||
| 1338 | return drm_debugfs_create_files(i915_debugfs_list, | 1448 | return drm_debugfs_create_files(i915_debugfs_list, |
| 1339 | I915_DEBUGFS_ENTRIES, | 1449 | I915_DEBUGFS_ENTRIES, |
| 1340 | minor->debugfs_root, minor); | 1450 | minor->debugfs_root, minor); |
| @@ -1344,6 +1454,8 @@ void i915_debugfs_cleanup(struct drm_minor *minor) | |||
| 1344 | { | 1454 | { |
| 1345 | drm_debugfs_remove_files(i915_debugfs_list, | 1455 | drm_debugfs_remove_files(i915_debugfs_list, |
| 1346 | I915_DEBUGFS_ENTRIES, minor); | 1456 | I915_DEBUGFS_ENTRIES, minor); |
| 1457 | drm_debugfs_remove_files((struct drm_info_list *) &i915_forcewake_fops, | ||
| 1458 | 1, minor); | ||
| 1347 | drm_debugfs_remove_files((struct drm_info_list *) &i915_wedged_fops, | 1459 | drm_debugfs_remove_files((struct drm_info_list *) &i915_wedged_fops, |
| 1348 | 1, minor); | 1460 | 1, minor); |
| 1349 | } | 1461 | } |
diff --git a/drivers/gpu/drm/i915/i915_dma.c b/drivers/gpu/drm/i915/i915_dma.c index 12876f2795d2..0239e9974bf2 100644 --- a/drivers/gpu/drm/i915/i915_dma.c +++ b/drivers/gpu/drm/i915/i915_dma.c | |||
| @@ -571,7 +571,7 @@ static int i915_quiescent(struct drm_device *dev) | |||
| 571 | struct intel_ring_buffer *ring = LP_RING(dev->dev_private); | 571 | struct intel_ring_buffer *ring = LP_RING(dev->dev_private); |
| 572 | 572 | ||
| 573 | i915_kernel_lost_context(dev); | 573 | i915_kernel_lost_context(dev); |
| 574 | return intel_wait_ring_buffer(ring, ring->size - 8); | 574 | return intel_wait_ring_idle(ring); |
| 575 | } | 575 | } |
| 576 | 576 | ||
| 577 | static int i915_flush_ioctl(struct drm_device *dev, void *data, | 577 | static int i915_flush_ioctl(struct drm_device *dev, void *data, |
| @@ -1176,11 +1176,11 @@ static bool i915_switcheroo_can_switch(struct pci_dev *pdev) | |||
| 1176 | return can_switch; | 1176 | return can_switch; |
| 1177 | } | 1177 | } |
| 1178 | 1178 | ||
| 1179 | static int i915_load_modeset_init(struct drm_device *dev) | 1179 | static int i915_load_gem_init(struct drm_device *dev) |
| 1180 | { | 1180 | { |
| 1181 | struct drm_i915_private *dev_priv = dev->dev_private; | 1181 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 1182 | unsigned long prealloc_size, gtt_size, mappable_size; | 1182 | unsigned long prealloc_size, gtt_size, mappable_size; |
| 1183 | int ret = 0; | 1183 | int ret; |
| 1184 | 1184 | ||
| 1185 | prealloc_size = dev_priv->mm.gtt->stolen_size; | 1185 | prealloc_size = dev_priv->mm.gtt->stolen_size; |
| 1186 | gtt_size = dev_priv->mm.gtt->gtt_total_entries << PAGE_SHIFT; | 1186 | gtt_size = dev_priv->mm.gtt->gtt_total_entries << PAGE_SHIFT; |
| @@ -1204,7 +1204,7 @@ static int i915_load_modeset_init(struct drm_device *dev) | |||
| 1204 | ret = i915_gem_init_ringbuffer(dev); | 1204 | ret = i915_gem_init_ringbuffer(dev); |
| 1205 | mutex_unlock(&dev->struct_mutex); | 1205 | mutex_unlock(&dev->struct_mutex); |
| 1206 | if (ret) | 1206 | if (ret) |
| 1207 | goto out; | 1207 | return ret; |
| 1208 | 1208 | ||
| 1209 | /* Try to set up FBC with a reasonable compressed buffer size */ | 1209 | /* Try to set up FBC with a reasonable compressed buffer size */ |
| 1210 | if (I915_HAS_FBC(dev) && i915_powersave) { | 1210 | if (I915_HAS_FBC(dev) && i915_powersave) { |
| @@ -1222,6 +1222,13 @@ static int i915_load_modeset_init(struct drm_device *dev) | |||
| 1222 | 1222 | ||
| 1223 | /* Allow hardware batchbuffers unless told otherwise. */ | 1223 | /* Allow hardware batchbuffers unless told otherwise. */ |
| 1224 | dev_priv->allow_batchbuffer = 1; | 1224 | dev_priv->allow_batchbuffer = 1; |
| 1225 | return 0; | ||
| 1226 | } | ||
| 1227 | |||
| 1228 | static int i915_load_modeset_init(struct drm_device *dev) | ||
| 1229 | { | ||
| 1230 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 1231 | int ret; | ||
| 1225 | 1232 | ||
| 1226 | ret = intel_parse_bios(dev); | 1233 | ret = intel_parse_bios(dev); |
| 1227 | if (ret) | 1234 | if (ret) |
| @@ -1236,7 +1243,7 @@ static int i915_load_modeset_init(struct drm_device *dev) | |||
| 1236 | */ | 1243 | */ |
| 1237 | ret = vga_client_register(dev->pdev, dev, NULL, i915_vga_set_decode); | 1244 | ret = vga_client_register(dev->pdev, dev, NULL, i915_vga_set_decode); |
| 1238 | if (ret && ret != -ENODEV) | 1245 | if (ret && ret != -ENODEV) |
| 1239 | goto cleanup_ringbuffer; | 1246 | goto out; |
| 1240 | 1247 | ||
| 1241 | intel_register_dsm_handler(); | 1248 | intel_register_dsm_handler(); |
| 1242 | 1249 | ||
| @@ -1253,10 +1260,40 @@ static int i915_load_modeset_init(struct drm_device *dev) | |||
| 1253 | 1260 | ||
| 1254 | intel_modeset_init(dev); | 1261 | intel_modeset_init(dev); |
| 1255 | 1262 | ||
| 1256 | ret = drm_irq_install(dev); | 1263 | ret = i915_load_gem_init(dev); |
| 1257 | if (ret) | 1264 | if (ret) |
| 1258 | goto cleanup_vga_switcheroo; | 1265 | goto cleanup_vga_switcheroo; |
| 1259 | 1266 | ||
| 1267 | intel_modeset_gem_init(dev); | ||
| 1268 | |||
| 1269 | if (IS_IVYBRIDGE(dev)) { | ||
| 1270 | /* Share pre & uninstall handlers with ILK/SNB */ | ||
| 1271 | dev->driver->irq_handler = ivybridge_irq_handler; | ||
| 1272 | dev->driver->irq_preinstall = ironlake_irq_preinstall; | ||
| 1273 | dev->driver->irq_postinstall = ivybridge_irq_postinstall; | ||
| 1274 | dev->driver->irq_uninstall = ironlake_irq_uninstall; | ||
| 1275 | dev->driver->enable_vblank = ivybridge_enable_vblank; | ||
| 1276 | dev->driver->disable_vblank = ivybridge_disable_vblank; | ||
| 1277 | } else if (HAS_PCH_SPLIT(dev)) { | ||
| 1278 | dev->driver->irq_handler = ironlake_irq_handler; | ||
| 1279 | dev->driver->irq_preinstall = ironlake_irq_preinstall; | ||
| 1280 | dev->driver->irq_postinstall = ironlake_irq_postinstall; | ||
| 1281 | dev->driver->irq_uninstall = ironlake_irq_uninstall; | ||
| 1282 | dev->driver->enable_vblank = ironlake_enable_vblank; | ||
| 1283 | dev->driver->disable_vblank = ironlake_disable_vblank; | ||
| 1284 | } else { | ||
| 1285 | dev->driver->irq_preinstall = i915_driver_irq_preinstall; | ||
| 1286 | dev->driver->irq_postinstall = i915_driver_irq_postinstall; | ||
| 1287 | dev->driver->irq_uninstall = i915_driver_irq_uninstall; | ||
| 1288 | dev->driver->irq_handler = i915_driver_irq_handler; | ||
| 1289 | dev->driver->enable_vblank = i915_enable_vblank; | ||
| 1290 | dev->driver->disable_vblank = i915_disable_vblank; | ||
| 1291 | } | ||
| 1292 | |||
| 1293 | ret = drm_irq_install(dev); | ||
| 1294 | if (ret) | ||
| 1295 | goto cleanup_gem; | ||
| 1296 | |||
| 1260 | /* Always safe in the mode setting case. */ | 1297 | /* Always safe in the mode setting case. */ |
| 1261 | /* FIXME: do pre/post-mode set stuff in core KMS code */ | 1298 | /* FIXME: do pre/post-mode set stuff in core KMS code */ |
| 1262 | dev->vblank_disable_allowed = 1; | 1299 | dev->vblank_disable_allowed = 1; |
| @@ -1274,14 +1311,14 @@ static int i915_load_modeset_init(struct drm_device *dev) | |||
| 1274 | 1311 | ||
| 1275 | cleanup_irq: | 1312 | cleanup_irq: |
| 1276 | drm_irq_uninstall(dev); | 1313 | drm_irq_uninstall(dev); |
| 1314 | cleanup_gem: | ||
| 1315 | mutex_lock(&dev->struct_mutex); | ||
| 1316 | i915_gem_cleanup_ringbuffer(dev); | ||
| 1317 | mutex_unlock(&dev->struct_mutex); | ||
| 1277 | cleanup_vga_switcheroo: | 1318 | cleanup_vga_switcheroo: |
| 1278 | vga_switcheroo_unregister_client(dev->pdev); | 1319 | vga_switcheroo_unregister_client(dev->pdev); |
| 1279 | cleanup_vga_client: | 1320 | cleanup_vga_client: |
| 1280 | vga_client_register(dev->pdev, NULL, NULL, NULL); | 1321 | vga_client_register(dev->pdev, NULL, NULL, NULL); |
| 1281 | cleanup_ringbuffer: | ||
| 1282 | mutex_lock(&dev->struct_mutex); | ||
| 1283 | i915_gem_cleanup_ringbuffer(dev); | ||
| 1284 | mutex_unlock(&dev->struct_mutex); | ||
| 1285 | out: | 1322 | out: |
| 1286 | return ret; | 1323 | return ret; |
| 1287 | } | 1324 | } |
| @@ -1982,7 +2019,7 @@ int i915_driver_load(struct drm_device *dev, unsigned long flags) | |||
| 1982 | 2019 | ||
| 1983 | dev->driver->get_vblank_counter = i915_get_vblank_counter; | 2020 | dev->driver->get_vblank_counter = i915_get_vblank_counter; |
| 1984 | dev->max_vblank_count = 0xffffff; /* only 24 bits of frame count */ | 2021 | dev->max_vblank_count = 0xffffff; /* only 24 bits of frame count */ |
| 1985 | if (IS_G4X(dev) || IS_GEN5(dev) || IS_GEN6(dev)) { | 2022 | if (IS_G4X(dev) || IS_GEN5(dev) || IS_GEN6(dev) || IS_IVYBRIDGE(dev)) { |
| 1986 | dev->max_vblank_count = 0xffffffff; /* full 32 bit counter */ | 2023 | dev->max_vblank_count = 0xffffffff; /* full 32 bit counter */ |
| 1987 | dev->driver->get_vblank_counter = gm45_get_vblank_counter; | 2024 | dev->driver->get_vblank_counter = gm45_get_vblank_counter; |
| 1988 | } | 2025 | } |
| @@ -2025,6 +2062,7 @@ int i915_driver_load(struct drm_device *dev, unsigned long flags) | |||
| 2025 | 2062 | ||
| 2026 | spin_lock_init(&dev_priv->irq_lock); | 2063 | spin_lock_init(&dev_priv->irq_lock); |
| 2027 | spin_lock_init(&dev_priv->error_lock); | 2064 | spin_lock_init(&dev_priv->error_lock); |
| 2065 | spin_lock_init(&dev_priv->rps_lock); | ||
| 2028 | 2066 | ||
| 2029 | if (IS_MOBILE(dev) || !IS_GEN2(dev)) | 2067 | if (IS_MOBILE(dev) || !IS_GEN2(dev)) |
| 2030 | dev_priv->num_pipe = 2; | 2068 | dev_priv->num_pipe = 2; |
diff --git a/drivers/gpu/drm/i915/i915_drv.c b/drivers/gpu/drm/i915/i915_drv.c index c34a8dd31d02..8c4fcbb8a4cb 100644 --- a/drivers/gpu/drm/i915/i915_drv.c +++ b/drivers/gpu/drm/i915/i915_drv.c | |||
| @@ -188,6 +188,21 @@ static const struct intel_device_info intel_sandybridge_m_info = { | |||
| 188 | .has_blt_ring = 1, | 188 | .has_blt_ring = 1, |
| 189 | }; | 189 | }; |
| 190 | 190 | ||
| 191 | static const struct intel_device_info intel_ivybridge_d_info = { | ||
| 192 | .is_ivybridge = 1, .gen = 7, | ||
| 193 | .need_gfx_hws = 1, .has_hotplug = 1, | ||
| 194 | .has_bsd_ring = 1, | ||
| 195 | .has_blt_ring = 1, | ||
| 196 | }; | ||
| 197 | |||
| 198 | static const struct intel_device_info intel_ivybridge_m_info = { | ||
| 199 | .is_ivybridge = 1, .gen = 7, .is_mobile = 1, | ||
| 200 | .need_gfx_hws = 1, .has_hotplug = 1, | ||
| 201 | .has_fbc = 0, /* FBC is not enabled on Ivybridge mobile yet */ | ||
| 202 | .has_bsd_ring = 1, | ||
| 203 | .has_blt_ring = 1, | ||
| 204 | }; | ||
| 205 | |||
| 191 | static const struct pci_device_id pciidlist[] = { /* aka */ | 206 | static const struct pci_device_id pciidlist[] = { /* aka */ |
| 192 | INTEL_VGA_DEVICE(0x3577, &intel_i830_info), /* I830_M */ | 207 | INTEL_VGA_DEVICE(0x3577, &intel_i830_info), /* I830_M */ |
| 193 | INTEL_VGA_DEVICE(0x2562, &intel_845g_info), /* 845_G */ | 208 | INTEL_VGA_DEVICE(0x2562, &intel_845g_info), /* 845_G */ |
| @@ -227,6 +242,11 @@ static const struct pci_device_id pciidlist[] = { /* aka */ | |||
| 227 | INTEL_VGA_DEVICE(0x0116, &intel_sandybridge_m_info), | 242 | INTEL_VGA_DEVICE(0x0116, &intel_sandybridge_m_info), |
| 228 | INTEL_VGA_DEVICE(0x0126, &intel_sandybridge_m_info), | 243 | INTEL_VGA_DEVICE(0x0126, &intel_sandybridge_m_info), |
| 229 | INTEL_VGA_DEVICE(0x010A, &intel_sandybridge_d_info), | 244 | INTEL_VGA_DEVICE(0x010A, &intel_sandybridge_d_info), |
| 245 | INTEL_VGA_DEVICE(0x0156, &intel_ivybridge_m_info), /* GT1 mobile */ | ||
| 246 | INTEL_VGA_DEVICE(0x0166, &intel_ivybridge_m_info), /* GT2 mobile */ | ||
| 247 | INTEL_VGA_DEVICE(0x0152, &intel_ivybridge_d_info), /* GT1 desktop */ | ||
| 248 | INTEL_VGA_DEVICE(0x0162, &intel_ivybridge_d_info), /* GT2 desktop */ | ||
| 249 | INTEL_VGA_DEVICE(0x015a, &intel_ivybridge_d_info), /* GT1 server */ | ||
| 230 | {0, 0, 0} | 250 | {0, 0, 0} |
| 231 | }; | 251 | }; |
| 232 | 252 | ||
| @@ -235,7 +255,9 @@ MODULE_DEVICE_TABLE(pci, pciidlist); | |||
| 235 | #endif | 255 | #endif |
| 236 | 256 | ||
| 237 | #define INTEL_PCH_DEVICE_ID_MASK 0xff00 | 257 | #define INTEL_PCH_DEVICE_ID_MASK 0xff00 |
| 258 | #define INTEL_PCH_IBX_DEVICE_ID_TYPE 0x3b00 | ||
| 238 | #define INTEL_PCH_CPT_DEVICE_ID_TYPE 0x1c00 | 259 | #define INTEL_PCH_CPT_DEVICE_ID_TYPE 0x1c00 |
| 260 | #define INTEL_PCH_PPT_DEVICE_ID_TYPE 0x1e00 | ||
| 239 | 261 | ||
| 240 | void intel_detect_pch (struct drm_device *dev) | 262 | void intel_detect_pch (struct drm_device *dev) |
| 241 | { | 263 | { |
| @@ -254,16 +276,23 @@ void intel_detect_pch (struct drm_device *dev) | |||
| 254 | int id; | 276 | int id; |
| 255 | id = pch->device & INTEL_PCH_DEVICE_ID_MASK; | 277 | id = pch->device & INTEL_PCH_DEVICE_ID_MASK; |
| 256 | 278 | ||
| 257 | if (id == INTEL_PCH_CPT_DEVICE_ID_TYPE) { | 279 | if (id == INTEL_PCH_IBX_DEVICE_ID_TYPE) { |
| 280 | dev_priv->pch_type = PCH_IBX; | ||
| 281 | DRM_DEBUG_KMS("Found Ibex Peak PCH\n"); | ||
| 282 | } else if (id == INTEL_PCH_CPT_DEVICE_ID_TYPE) { | ||
| 258 | dev_priv->pch_type = PCH_CPT; | 283 | dev_priv->pch_type = PCH_CPT; |
| 259 | DRM_DEBUG_KMS("Found CougarPoint PCH\n"); | 284 | DRM_DEBUG_KMS("Found CougarPoint PCH\n"); |
| 285 | } else if (id == INTEL_PCH_PPT_DEVICE_ID_TYPE) { | ||
| 286 | /* PantherPoint is CPT compatible */ | ||
| 287 | dev_priv->pch_type = PCH_CPT; | ||
| 288 | DRM_DEBUG_KMS("Found PatherPoint PCH\n"); | ||
| 260 | } | 289 | } |
| 261 | } | 290 | } |
| 262 | pci_dev_put(pch); | 291 | pci_dev_put(pch); |
| 263 | } | 292 | } |
| 264 | } | 293 | } |
| 265 | 294 | ||
| 266 | void __gen6_gt_force_wake_get(struct drm_i915_private *dev_priv) | 295 | static void __gen6_gt_force_wake_get(struct drm_i915_private *dev_priv) |
| 267 | { | 296 | { |
| 268 | int count; | 297 | int count; |
| 269 | 298 | ||
| @@ -279,12 +308,38 @@ void __gen6_gt_force_wake_get(struct drm_i915_private *dev_priv) | |||
| 279 | udelay(10); | 308 | udelay(10); |
| 280 | } | 309 | } |
| 281 | 310 | ||
| 282 | void __gen6_gt_force_wake_put(struct drm_i915_private *dev_priv) | 311 | /* |
| 312 | * Generally this is called implicitly by the register read function. However, | ||
| 313 | * if some sequence requires the GT to not power down then this function should | ||
| 314 | * be called at the beginning of the sequence followed by a call to | ||
| 315 | * gen6_gt_force_wake_put() at the end of the sequence. | ||
| 316 | */ | ||
| 317 | void gen6_gt_force_wake_get(struct drm_i915_private *dev_priv) | ||
| 318 | { | ||
| 319 | WARN_ON(!mutex_is_locked(&dev_priv->dev->struct_mutex)); | ||
| 320 | |||
| 321 | /* Forcewake is atomic in case we get in here without the lock */ | ||
| 322 | if (atomic_add_return(1, &dev_priv->forcewake_count) == 1) | ||
| 323 | __gen6_gt_force_wake_get(dev_priv); | ||
| 324 | } | ||
| 325 | |||
| 326 | static void __gen6_gt_force_wake_put(struct drm_i915_private *dev_priv) | ||
| 283 | { | 327 | { |
| 284 | I915_WRITE_NOTRACE(FORCEWAKE, 0); | 328 | I915_WRITE_NOTRACE(FORCEWAKE, 0); |
| 285 | POSTING_READ(FORCEWAKE); | 329 | POSTING_READ(FORCEWAKE); |
| 286 | } | 330 | } |
| 287 | 331 | ||
| 332 | /* | ||
| 333 | * see gen6_gt_force_wake_get() | ||
| 334 | */ | ||
| 335 | void gen6_gt_force_wake_put(struct drm_i915_private *dev_priv) | ||
| 336 | { | ||
| 337 | WARN_ON(!mutex_is_locked(&dev_priv->dev->struct_mutex)); | ||
| 338 | |||
| 339 | if (atomic_dec_and_test(&dev_priv->forcewake_count)) | ||
| 340 | __gen6_gt_force_wake_put(dev_priv); | ||
| 341 | } | ||
| 342 | |||
| 288 | void __gen6_gt_wait_for_fifo(struct drm_i915_private *dev_priv) | 343 | void __gen6_gt_wait_for_fifo(struct drm_i915_private *dev_priv) |
| 289 | { | 344 | { |
| 290 | int loop = 500; | 345 | int loop = 500; |
diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h index 1c1b27c97e5c..3a1c27718065 100644 --- a/drivers/gpu/drm/i915/i915_drv.h +++ b/drivers/gpu/drm/i915/i915_drv.h | |||
| @@ -188,7 +188,7 @@ struct drm_i915_error_state { | |||
| 188 | u32 dirty:1; | 188 | u32 dirty:1; |
| 189 | u32 purgeable:1; | 189 | u32 purgeable:1; |
| 190 | u32 ring:4; | 190 | u32 ring:4; |
| 191 | u32 agp_type:1; | 191 | u32 cache_level:2; |
| 192 | } *active_bo, *pinned_bo; | 192 | } *active_bo, *pinned_bo; |
| 193 | u32 active_bo_count, pinned_bo_count; | 193 | u32 active_bo_count, pinned_bo_count; |
| 194 | struct intel_overlay_error_state *overlay; | 194 | struct intel_overlay_error_state *overlay; |
| @@ -203,12 +203,19 @@ struct drm_i915_display_funcs { | |||
| 203 | int (*get_display_clock_speed)(struct drm_device *dev); | 203 | int (*get_display_clock_speed)(struct drm_device *dev); |
| 204 | int (*get_fifo_size)(struct drm_device *dev, int plane); | 204 | int (*get_fifo_size)(struct drm_device *dev, int plane); |
| 205 | void (*update_wm)(struct drm_device *dev); | 205 | void (*update_wm)(struct drm_device *dev); |
| 206 | int (*crtc_mode_set)(struct drm_crtc *crtc, | ||
| 207 | struct drm_display_mode *mode, | ||
| 208 | struct drm_display_mode *adjusted_mode, | ||
| 209 | int x, int y, | ||
| 210 | struct drm_framebuffer *old_fb); | ||
| 211 | void (*fdi_link_train)(struct drm_crtc *crtc); | ||
| 212 | void (*init_clock_gating)(struct drm_device *dev); | ||
| 213 | void (*init_pch_clock_gating)(struct drm_device *dev); | ||
| 206 | /* clock updates for mode set */ | 214 | /* clock updates for mode set */ |
| 207 | /* cursor updates */ | 215 | /* cursor updates */ |
| 208 | /* render clock increase/decrease */ | 216 | /* render clock increase/decrease */ |
| 209 | /* display clock increase/decrease */ | 217 | /* display clock increase/decrease */ |
| 210 | /* pll clock increase/decrease */ | 218 | /* pll clock increase/decrease */ |
| 211 | /* clock gating init */ | ||
| 212 | }; | 219 | }; |
| 213 | 220 | ||
| 214 | struct intel_device_info { | 221 | struct intel_device_info { |
| @@ -223,6 +230,7 @@ struct intel_device_info { | |||
| 223 | u8 is_pineview : 1; | 230 | u8 is_pineview : 1; |
| 224 | u8 is_broadwater : 1; | 231 | u8 is_broadwater : 1; |
| 225 | u8 is_crestline : 1; | 232 | u8 is_crestline : 1; |
| 233 | u8 is_ivybridge : 1; | ||
| 226 | u8 has_fbc : 1; | 234 | u8 has_fbc : 1; |
| 227 | u8 has_pipe_cxsr : 1; | 235 | u8 has_pipe_cxsr : 1; |
| 228 | u8 has_hotplug : 1; | 236 | u8 has_hotplug : 1; |
| @@ -676,6 +684,10 @@ typedef struct drm_i915_private { | |||
| 676 | 684 | ||
| 677 | bool mchbar_need_disable; | 685 | bool mchbar_need_disable; |
| 678 | 686 | ||
| 687 | struct work_struct rps_work; | ||
| 688 | spinlock_t rps_lock; | ||
| 689 | u32 pm_iir; | ||
| 690 | |||
| 679 | u8 cur_delay; | 691 | u8 cur_delay; |
| 680 | u8 min_delay; | 692 | u8 min_delay; |
| 681 | u8 max_delay; | 693 | u8 max_delay; |
| @@ -703,8 +715,16 @@ typedef struct drm_i915_private { | |||
| 703 | struct intel_fbdev *fbdev; | 715 | struct intel_fbdev *fbdev; |
| 704 | 716 | ||
| 705 | struct drm_property *broadcast_rgb_property; | 717 | struct drm_property *broadcast_rgb_property; |
| 718 | |||
| 719 | atomic_t forcewake_count; | ||
| 706 | } drm_i915_private_t; | 720 | } drm_i915_private_t; |
| 707 | 721 | ||
| 722 | enum i915_cache_level { | ||
| 723 | I915_CACHE_NONE, | ||
| 724 | I915_CACHE_LLC, | ||
| 725 | I915_CACHE_LLC_MLC, /* gen6+ */ | ||
| 726 | }; | ||
| 727 | |||
| 708 | struct drm_i915_gem_object { | 728 | struct drm_i915_gem_object { |
| 709 | struct drm_gem_object base; | 729 | struct drm_gem_object base; |
| 710 | 730 | ||
| @@ -791,6 +811,8 @@ struct drm_i915_gem_object { | |||
| 791 | unsigned int pending_fenced_gpu_access:1; | 811 | unsigned int pending_fenced_gpu_access:1; |
| 792 | unsigned int fenced_gpu_access:1; | 812 | unsigned int fenced_gpu_access:1; |
| 793 | 813 | ||
| 814 | unsigned int cache_level:2; | ||
| 815 | |||
| 794 | struct page **pages; | 816 | struct page **pages; |
| 795 | 817 | ||
| 796 | /** | 818 | /** |
| @@ -827,8 +849,6 @@ struct drm_i915_gem_object { | |||
| 827 | /** Record of address bit 17 of each page at last unbind. */ | 849 | /** Record of address bit 17 of each page at last unbind. */ |
| 828 | unsigned long *bit_17; | 850 | unsigned long *bit_17; |
| 829 | 851 | ||
| 830 | /** AGP mapping type (AGP_USER_MEMORY or AGP_USER_CACHED_MEMORY */ | ||
| 831 | uint32_t agp_type; | ||
| 832 | 852 | ||
| 833 | /** | 853 | /** |
| 834 | * If present, while GEM_DOMAIN_CPU is in the read domain this array | 854 | * If present, while GEM_DOMAIN_CPU is in the read domain this array |
| @@ -915,13 +935,21 @@ enum intel_chip_family { | |||
| 915 | #define IS_G33(dev) (INTEL_INFO(dev)->is_g33) | 935 | #define IS_G33(dev) (INTEL_INFO(dev)->is_g33) |
| 916 | #define IS_IRONLAKE_D(dev) ((dev)->pci_device == 0x0042) | 936 | #define IS_IRONLAKE_D(dev) ((dev)->pci_device == 0x0042) |
| 917 | #define IS_IRONLAKE_M(dev) ((dev)->pci_device == 0x0046) | 937 | #define IS_IRONLAKE_M(dev) ((dev)->pci_device == 0x0046) |
| 938 | #define IS_IVYBRIDGE(dev) (INTEL_INFO(dev)->is_ivybridge) | ||
| 918 | #define IS_MOBILE(dev) (INTEL_INFO(dev)->is_mobile) | 939 | #define IS_MOBILE(dev) (INTEL_INFO(dev)->is_mobile) |
| 919 | 940 | ||
| 941 | /* | ||
| 942 | * The genX designation typically refers to the render engine, so render | ||
| 943 | * capability related checks should use IS_GEN, while display and other checks | ||
| 944 | * have their own (e.g. HAS_PCH_SPLIT for ILK+ display, IS_foo for particular | ||
| 945 | * chips, etc.). | ||
| 946 | */ | ||
| 920 | #define IS_GEN2(dev) (INTEL_INFO(dev)->gen == 2) | 947 | #define IS_GEN2(dev) (INTEL_INFO(dev)->gen == 2) |
| 921 | #define IS_GEN3(dev) (INTEL_INFO(dev)->gen == 3) | 948 | #define IS_GEN3(dev) (INTEL_INFO(dev)->gen == 3) |
| 922 | #define IS_GEN4(dev) (INTEL_INFO(dev)->gen == 4) | 949 | #define IS_GEN4(dev) (INTEL_INFO(dev)->gen == 4) |
| 923 | #define IS_GEN5(dev) (INTEL_INFO(dev)->gen == 5) | 950 | #define IS_GEN5(dev) (INTEL_INFO(dev)->gen == 5) |
| 924 | #define IS_GEN6(dev) (INTEL_INFO(dev)->gen == 6) | 951 | #define IS_GEN6(dev) (INTEL_INFO(dev)->gen == 6) |
| 952 | #define IS_GEN7(dev) (INTEL_INFO(dev)->gen == 7) | ||
| 925 | 953 | ||
| 926 | #define HAS_BSD(dev) (INTEL_INFO(dev)->has_bsd_ring) | 954 | #define HAS_BSD(dev) (INTEL_INFO(dev)->has_bsd_ring) |
| 927 | #define HAS_BLT(dev) (INTEL_INFO(dev)->has_blt_ring) | 955 | #define HAS_BLT(dev) (INTEL_INFO(dev)->has_blt_ring) |
| @@ -948,8 +976,8 @@ enum intel_chip_family { | |||
| 948 | #define HAS_PIPE_CXSR(dev) (INTEL_INFO(dev)->has_pipe_cxsr) | 976 | #define HAS_PIPE_CXSR(dev) (INTEL_INFO(dev)->has_pipe_cxsr) |
| 949 | #define I915_HAS_FBC(dev) (INTEL_INFO(dev)->has_fbc) | 977 | #define I915_HAS_FBC(dev) (INTEL_INFO(dev)->has_fbc) |
| 950 | 978 | ||
| 951 | #define HAS_PCH_SPLIT(dev) (IS_GEN5(dev) || IS_GEN6(dev)) | 979 | #define HAS_PCH_SPLIT(dev) (IS_GEN5(dev) || IS_GEN6(dev) || IS_IVYBRIDGE(dev)) |
| 952 | #define HAS_PIPE_CONTROL(dev) (IS_GEN5(dev) || IS_GEN6(dev)) | 980 | #define HAS_PIPE_CONTROL(dev) (INTEL_INFO(dev)->gen >= 5) |
| 953 | 981 | ||
| 954 | #define INTEL_PCH_TYPE(dev) (((struct drm_i915_private *)(dev)->dev_private)->pch_type) | 982 | #define INTEL_PCH_TYPE(dev) (((struct drm_i915_private *)(dev)->dev_private)->pch_type) |
| 955 | #define HAS_PCH_CPT(dev) (INTEL_PCH_TYPE(dev) == PCH_CPT) | 983 | #define HAS_PCH_CPT(dev) (INTEL_PCH_TYPE(dev) == PCH_CPT) |
| @@ -1010,12 +1038,27 @@ extern irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS); | |||
| 1010 | extern void i915_driver_irq_preinstall(struct drm_device * dev); | 1038 | extern void i915_driver_irq_preinstall(struct drm_device * dev); |
| 1011 | extern int i915_driver_irq_postinstall(struct drm_device *dev); | 1039 | extern int i915_driver_irq_postinstall(struct drm_device *dev); |
| 1012 | extern void i915_driver_irq_uninstall(struct drm_device * dev); | 1040 | extern void i915_driver_irq_uninstall(struct drm_device * dev); |
| 1041 | |||
| 1042 | extern irqreturn_t ironlake_irq_handler(DRM_IRQ_ARGS); | ||
| 1043 | extern void ironlake_irq_preinstall(struct drm_device *dev); | ||
| 1044 | extern int ironlake_irq_postinstall(struct drm_device *dev); | ||
| 1045 | extern void ironlake_irq_uninstall(struct drm_device *dev); | ||
| 1046 | |||
| 1047 | extern irqreturn_t ivybridge_irq_handler(DRM_IRQ_ARGS); | ||
| 1048 | extern void ivybridge_irq_preinstall(struct drm_device *dev); | ||
| 1049 | extern int ivybridge_irq_postinstall(struct drm_device *dev); | ||
| 1050 | extern void ivybridge_irq_uninstall(struct drm_device *dev); | ||
| 1051 | |||
| 1013 | extern int i915_vblank_pipe_set(struct drm_device *dev, void *data, | 1052 | extern int i915_vblank_pipe_set(struct drm_device *dev, void *data, |
| 1014 | struct drm_file *file_priv); | 1053 | struct drm_file *file_priv); |
| 1015 | extern int i915_vblank_pipe_get(struct drm_device *dev, void *data, | 1054 | extern int i915_vblank_pipe_get(struct drm_device *dev, void *data, |
| 1016 | struct drm_file *file_priv); | 1055 | struct drm_file *file_priv); |
| 1017 | extern int i915_enable_vblank(struct drm_device *dev, int crtc); | 1056 | extern int i915_enable_vblank(struct drm_device *dev, int crtc); |
| 1018 | extern void i915_disable_vblank(struct drm_device *dev, int crtc); | 1057 | extern void i915_disable_vblank(struct drm_device *dev, int crtc); |
| 1058 | extern int ironlake_enable_vblank(struct drm_device *dev, int crtc); | ||
| 1059 | extern void ironlake_disable_vblank(struct drm_device *dev, int crtc); | ||
| 1060 | extern int ivybridge_enable_vblank(struct drm_device *dev, int crtc); | ||
| 1061 | extern void ivybridge_disable_vblank(struct drm_device *dev, int crtc); | ||
| 1019 | extern u32 i915_get_vblank_counter(struct drm_device *dev, int crtc); | 1062 | extern u32 i915_get_vblank_counter(struct drm_device *dev, int crtc); |
| 1020 | extern u32 gm45_get_vblank_counter(struct drm_device *dev, int crtc); | 1063 | extern u32 gm45_get_vblank_counter(struct drm_device *dev, int crtc); |
| 1021 | extern int i915_vblank_swap(struct drm_device *dev, void *data, | 1064 | extern int i915_vblank_swap(struct drm_device *dev, void *data, |
| @@ -1265,6 +1308,7 @@ static inline void intel_unregister_dsm_handler(void) { return; } | |||
| 1265 | 1308 | ||
| 1266 | /* modesetting */ | 1309 | /* modesetting */ |
| 1267 | extern void intel_modeset_init(struct drm_device *dev); | 1310 | extern void intel_modeset_init(struct drm_device *dev); |
| 1311 | extern void intel_modeset_gem_init(struct drm_device *dev); | ||
| 1268 | extern void intel_modeset_cleanup(struct drm_device *dev); | 1312 | extern void intel_modeset_cleanup(struct drm_device *dev); |
| 1269 | extern int intel_modeset_vga_set_state(struct drm_device *dev, bool state); | 1313 | extern int intel_modeset_vga_set_state(struct drm_device *dev, bool state); |
| 1270 | extern void i8xx_disable_fbc(struct drm_device *dev); | 1314 | extern void i8xx_disable_fbc(struct drm_device *dev); |
| @@ -1312,13 +1356,34 @@ extern void intel_display_print_error_state(struct seq_file *m, | |||
| 1312 | LOCK_TEST_WITH_RETURN(dev, file); \ | 1356 | LOCK_TEST_WITH_RETURN(dev, file); \ |
| 1313 | } while (0) | 1357 | } while (0) |
| 1314 | 1358 | ||
| 1359 | /* On SNB platform, before reading ring registers forcewake bit | ||
| 1360 | * must be set to prevent GT core from power down and stale values being | ||
| 1361 | * returned. | ||
| 1362 | */ | ||
| 1363 | void gen6_gt_force_wake_get(struct drm_i915_private *dev_priv); | ||
| 1364 | void gen6_gt_force_wake_put(struct drm_i915_private *dev_priv); | ||
| 1365 | void __gen6_gt_wait_for_fifo(struct drm_i915_private *dev_priv); | ||
| 1366 | |||
| 1367 | /* We give fast paths for the really cool registers */ | ||
| 1368 | #define NEEDS_FORCE_WAKE(dev_priv, reg) \ | ||
| 1369 | (((dev_priv)->info->gen >= 6) && \ | ||
| 1370 | ((reg) < 0x40000) && \ | ||
| 1371 | ((reg) != FORCEWAKE)) | ||
| 1315 | 1372 | ||
| 1316 | #define __i915_read(x, y) \ | 1373 | #define __i915_read(x, y) \ |
| 1317 | static inline u##x i915_read##x(struct drm_i915_private *dev_priv, u32 reg) { \ | 1374 | static inline u##x i915_read##x(struct drm_i915_private *dev_priv, u32 reg) { \ |
| 1318 | u##x val = read##y(dev_priv->regs + reg); \ | 1375 | u##x val = 0; \ |
| 1376 | if (NEEDS_FORCE_WAKE((dev_priv), (reg))) { \ | ||
| 1377 | gen6_gt_force_wake_get(dev_priv); \ | ||
| 1378 | val = read##y(dev_priv->regs + reg); \ | ||
| 1379 | gen6_gt_force_wake_put(dev_priv); \ | ||
| 1380 | } else { \ | ||
| 1381 | val = read##y(dev_priv->regs + reg); \ | ||
| 1382 | } \ | ||
| 1319 | trace_i915_reg_rw(false, reg, val, sizeof(val)); \ | 1383 | trace_i915_reg_rw(false, reg, val, sizeof(val)); \ |
| 1320 | return val; \ | 1384 | return val; \ |
| 1321 | } | 1385 | } |
| 1386 | |||
| 1322 | __i915_read(8, b) | 1387 | __i915_read(8, b) |
| 1323 | __i915_read(16, w) | 1388 | __i915_read(16, w) |
| 1324 | __i915_read(32, l) | 1389 | __i915_read(32, l) |
| @@ -1328,6 +1393,9 @@ __i915_read(64, q) | |||
| 1328 | #define __i915_write(x, y) \ | 1393 | #define __i915_write(x, y) \ |
| 1329 | static inline void i915_write##x(struct drm_i915_private *dev_priv, u32 reg, u##x val) { \ | 1394 | static inline void i915_write##x(struct drm_i915_private *dev_priv, u32 reg, u##x val) { \ |
| 1330 | trace_i915_reg_rw(true, reg, val, sizeof(val)); \ | 1395 | trace_i915_reg_rw(true, reg, val, sizeof(val)); \ |
| 1396 | if (NEEDS_FORCE_WAKE((dev_priv), (reg))) { \ | ||
| 1397 | __gen6_gt_wait_for_fifo(dev_priv); \ | ||
| 1398 | } \ | ||
| 1331 | write##y(val, dev_priv->regs + reg); \ | 1399 | write##y(val, dev_priv->regs + reg); \ |
| 1332 | } | 1400 | } |
| 1333 | __i915_write(8, b) | 1401 | __i915_write(8, b) |
| @@ -1356,33 +1424,4 @@ __i915_write(64, q) | |||
| 1356 | #define POSTING_READ16(reg) (void)I915_READ16_NOTRACE(reg) | 1424 | #define POSTING_READ16(reg) (void)I915_READ16_NOTRACE(reg) |
| 1357 | 1425 | ||
| 1358 | 1426 | ||
| 1359 | /* On SNB platform, before reading ring registers forcewake bit | ||
| 1360 | * must be set to prevent GT core from power down and stale values being | ||
| 1361 | * returned. | ||
| 1362 | */ | ||
| 1363 | void __gen6_gt_force_wake_get(struct drm_i915_private *dev_priv); | ||
| 1364 | void __gen6_gt_force_wake_put(struct drm_i915_private *dev_priv); | ||
| 1365 | void __gen6_gt_wait_for_fifo(struct drm_i915_private *dev_priv); | ||
| 1366 | |||
| 1367 | static inline u32 i915_gt_read(struct drm_i915_private *dev_priv, u32 reg) | ||
| 1368 | { | ||
| 1369 | u32 val; | ||
| 1370 | |||
| 1371 | if (dev_priv->info->gen >= 6) { | ||
| 1372 | __gen6_gt_force_wake_get(dev_priv); | ||
| 1373 | val = I915_READ(reg); | ||
| 1374 | __gen6_gt_force_wake_put(dev_priv); | ||
| 1375 | } else | ||
| 1376 | val = I915_READ(reg); | ||
| 1377 | |||
| 1378 | return val; | ||
| 1379 | } | ||
| 1380 | |||
| 1381 | static inline void i915_gt_write(struct drm_i915_private *dev_priv, | ||
| 1382 | u32 reg, u32 val) | ||
| 1383 | { | ||
| 1384 | if (dev_priv->info->gen >= 6) | ||
| 1385 | __gen6_gt_wait_for_fifo(dev_priv); | ||
| 1386 | I915_WRITE(reg, val); | ||
| 1387 | } | ||
| 1388 | #endif | 1427 | #endif |
diff --git a/drivers/gpu/drm/i915/i915_gem.c b/drivers/gpu/drm/i915/i915_gem.c index 7ce3f353af33..c6289034e29a 100644 --- a/drivers/gpu/drm/i915/i915_gem.c +++ b/drivers/gpu/drm/i915/i915_gem.c | |||
| @@ -2673,6 +2673,7 @@ i915_gem_object_get_fence(struct drm_i915_gem_object *obj, | |||
| 2673 | update: | 2673 | update: |
| 2674 | obj->tiling_changed = false; | 2674 | obj->tiling_changed = false; |
| 2675 | switch (INTEL_INFO(dev)->gen) { | 2675 | switch (INTEL_INFO(dev)->gen) { |
| 2676 | case 7: | ||
| 2676 | case 6: | 2677 | case 6: |
| 2677 | ret = sandybridge_write_fence_reg(obj, pipelined); | 2678 | ret = sandybridge_write_fence_reg(obj, pipelined); |
| 2678 | break; | 2679 | break; |
| @@ -2706,6 +2707,7 @@ i915_gem_clear_fence_reg(struct drm_device *dev, | |||
| 2706 | uint32_t fence_reg = reg - dev_priv->fence_regs; | 2707 | uint32_t fence_reg = reg - dev_priv->fence_regs; |
| 2707 | 2708 | ||
| 2708 | switch (INTEL_INFO(dev)->gen) { | 2709 | switch (INTEL_INFO(dev)->gen) { |
| 2710 | case 7: | ||
| 2709 | case 6: | 2711 | case 6: |
| 2710 | I915_WRITE64(FENCE_REG_SANDYBRIDGE_0 + fence_reg*8, 0); | 2712 | I915_WRITE64(FENCE_REG_SANDYBRIDGE_0 + fence_reg*8, 0); |
| 2711 | break; | 2713 | break; |
| @@ -2878,6 +2880,17 @@ i915_gem_clflush_object(struct drm_i915_gem_object *obj) | |||
| 2878 | if (obj->pages == NULL) | 2880 | if (obj->pages == NULL) |
| 2879 | return; | 2881 | return; |
| 2880 | 2882 | ||
| 2883 | /* If the GPU is snooping the contents of the CPU cache, | ||
| 2884 | * we do not need to manually clear the CPU cache lines. However, | ||
| 2885 | * the caches are only snooped when the render cache is | ||
| 2886 | * flushed/invalidated. As we always have to emit invalidations | ||
| 2887 | * and flushes when moving into and out of the RENDER domain, correct | ||
| 2888 | * snooping behaviour occurs naturally as the result of our domain | ||
| 2889 | * tracking. | ||
| 2890 | */ | ||
| 2891 | if (obj->cache_level != I915_CACHE_NONE) | ||
| 2892 | return; | ||
| 2893 | |||
| 2881 | trace_i915_gem_object_clflush(obj); | 2894 | trace_i915_gem_object_clflush(obj); |
| 2882 | 2895 | ||
| 2883 | drm_clflush_pages(obj->pages, obj->base.size / PAGE_SIZE); | 2896 | drm_clflush_pages(obj->pages, obj->base.size / PAGE_SIZE); |
| @@ -3569,7 +3582,7 @@ struct drm_i915_gem_object *i915_gem_alloc_object(struct drm_device *dev, | |||
| 3569 | obj->base.write_domain = I915_GEM_DOMAIN_CPU; | 3582 | obj->base.write_domain = I915_GEM_DOMAIN_CPU; |
| 3570 | obj->base.read_domains = I915_GEM_DOMAIN_CPU; | 3583 | obj->base.read_domains = I915_GEM_DOMAIN_CPU; |
| 3571 | 3584 | ||
| 3572 | obj->agp_type = AGP_USER_MEMORY; | 3585 | obj->cache_level = I915_CACHE_NONE; |
| 3573 | obj->base.driver_private = NULL; | 3586 | obj->base.driver_private = NULL; |
| 3574 | obj->fence_reg = I915_FENCE_REG_NONE; | 3587 | obj->fence_reg = I915_FENCE_REG_NONE; |
| 3575 | INIT_LIST_HEAD(&obj->mm_list); | 3588 | INIT_LIST_HEAD(&obj->mm_list); |
| @@ -3845,25 +3858,10 @@ i915_gem_load(struct drm_device *dev) | |||
| 3845 | dev_priv->num_fence_regs = 8; | 3858 | dev_priv->num_fence_regs = 8; |
| 3846 | 3859 | ||
| 3847 | /* Initialize fence registers to zero */ | 3860 | /* Initialize fence registers to zero */ |
| 3848 | switch (INTEL_INFO(dev)->gen) { | 3861 | for (i = 0; i < dev_priv->num_fence_regs; i++) { |
| 3849 | case 6: | 3862 | i915_gem_clear_fence_reg(dev, &dev_priv->fence_regs[i]); |
| 3850 | for (i = 0; i < 16; i++) | ||
| 3851 | I915_WRITE64(FENCE_REG_SANDYBRIDGE_0 + (i * 8), 0); | ||
| 3852 | break; | ||
| 3853 | case 5: | ||
| 3854 | case 4: | ||
| 3855 | for (i = 0; i < 16; i++) | ||
| 3856 | I915_WRITE64(FENCE_REG_965_0 + (i * 8), 0); | ||
| 3857 | break; | ||
| 3858 | case 3: | ||
| 3859 | if (IS_I945G(dev) || IS_I945GM(dev) || IS_G33(dev)) | ||
| 3860 | for (i = 0; i < 8; i++) | ||
| 3861 | I915_WRITE(FENCE_REG_945_8 + (i * 4), 0); | ||
| 3862 | case 2: | ||
| 3863 | for (i = 0; i < 8; i++) | ||
| 3864 | I915_WRITE(FENCE_REG_830_0 + (i * 4), 0); | ||
| 3865 | break; | ||
| 3866 | } | 3863 | } |
| 3864 | |||
| 3867 | i915_gem_detect_bit_6_swizzle(dev); | 3865 | i915_gem_detect_bit_6_swizzle(dev); |
| 3868 | init_waitqueue_head(&dev_priv->pending_flip_queue); | 3866 | init_waitqueue_head(&dev_priv->pending_flip_queue); |
| 3869 | 3867 | ||
diff --git a/drivers/gpu/drm/i915/i915_gem_gtt.c b/drivers/gpu/drm/i915/i915_gem_gtt.c index b0abdc64aa9f..e46b645773cf 100644 --- a/drivers/gpu/drm/i915/i915_gem_gtt.c +++ b/drivers/gpu/drm/i915/i915_gem_gtt.c | |||
| @@ -29,6 +29,26 @@ | |||
| 29 | #include "i915_trace.h" | 29 | #include "i915_trace.h" |
| 30 | #include "intel_drv.h" | 30 | #include "intel_drv.h" |
| 31 | 31 | ||
| 32 | /* XXX kill agp_type! */ | ||
| 33 | static unsigned int cache_level_to_agp_type(struct drm_device *dev, | ||
| 34 | enum i915_cache_level cache_level) | ||
| 35 | { | ||
| 36 | switch (cache_level) { | ||
| 37 | case I915_CACHE_LLC_MLC: | ||
| 38 | if (INTEL_INFO(dev)->gen >= 6) | ||
| 39 | return AGP_USER_CACHED_MEMORY_LLC_MLC; | ||
| 40 | /* Older chipsets do not have this extra level of CPU | ||
| 41 | * cacheing, so fallthrough and request the PTE simply | ||
| 42 | * as cached. | ||
| 43 | */ | ||
| 44 | case I915_CACHE_LLC: | ||
| 45 | return AGP_USER_CACHED_MEMORY; | ||
| 46 | default: | ||
| 47 | case I915_CACHE_NONE: | ||
| 48 | return AGP_USER_MEMORY; | ||
| 49 | } | ||
| 50 | } | ||
| 51 | |||
| 32 | void i915_gem_restore_gtt_mappings(struct drm_device *dev) | 52 | void i915_gem_restore_gtt_mappings(struct drm_device *dev) |
| 33 | { | 53 | { |
| 34 | struct drm_i915_private *dev_priv = dev->dev_private; | 54 | struct drm_i915_private *dev_priv = dev->dev_private; |
| @@ -39,6 +59,9 @@ void i915_gem_restore_gtt_mappings(struct drm_device *dev) | |||
| 39 | (dev_priv->mm.gtt_end - dev_priv->mm.gtt_start) / PAGE_SIZE); | 59 | (dev_priv->mm.gtt_end - dev_priv->mm.gtt_start) / PAGE_SIZE); |
| 40 | 60 | ||
| 41 | list_for_each_entry(obj, &dev_priv->mm.gtt_list, gtt_list) { | 61 | list_for_each_entry(obj, &dev_priv->mm.gtt_list, gtt_list) { |
| 62 | unsigned int agp_type = | ||
| 63 | cache_level_to_agp_type(dev, obj->cache_level); | ||
| 64 | |||
| 42 | i915_gem_clflush_object(obj); | 65 | i915_gem_clflush_object(obj); |
| 43 | 66 | ||
| 44 | if (dev_priv->mm.gtt->needs_dmar) { | 67 | if (dev_priv->mm.gtt->needs_dmar) { |
| @@ -46,15 +69,14 @@ void i915_gem_restore_gtt_mappings(struct drm_device *dev) | |||
| 46 | 69 | ||
| 47 | intel_gtt_insert_sg_entries(obj->sg_list, | 70 | intel_gtt_insert_sg_entries(obj->sg_list, |
| 48 | obj->num_sg, | 71 | obj->num_sg, |
| 49 | obj->gtt_space->start | 72 | obj->gtt_space->start >> PAGE_SHIFT, |
| 50 | >> PAGE_SHIFT, | 73 | agp_type); |
| 51 | obj->agp_type); | ||
| 52 | } else | 74 | } else |
| 53 | intel_gtt_insert_pages(obj->gtt_space->start | 75 | intel_gtt_insert_pages(obj->gtt_space->start |
| 54 | >> PAGE_SHIFT, | 76 | >> PAGE_SHIFT, |
| 55 | obj->base.size >> PAGE_SHIFT, | 77 | obj->base.size >> PAGE_SHIFT, |
| 56 | obj->pages, | 78 | obj->pages, |
| 57 | obj->agp_type); | 79 | agp_type); |
| 58 | } | 80 | } |
| 59 | 81 | ||
| 60 | intel_gtt_chipset_flush(); | 82 | intel_gtt_chipset_flush(); |
| @@ -64,6 +86,7 @@ int i915_gem_gtt_bind_object(struct drm_i915_gem_object *obj) | |||
| 64 | { | 86 | { |
| 65 | struct drm_device *dev = obj->base.dev; | 87 | struct drm_device *dev = obj->base.dev; |
| 66 | struct drm_i915_private *dev_priv = dev->dev_private; | 88 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 89 | unsigned int agp_type = cache_level_to_agp_type(dev, obj->cache_level); | ||
| 67 | int ret; | 90 | int ret; |
| 68 | 91 | ||
| 69 | if (dev_priv->mm.gtt->needs_dmar) { | 92 | if (dev_priv->mm.gtt->needs_dmar) { |
| @@ -77,12 +100,12 @@ int i915_gem_gtt_bind_object(struct drm_i915_gem_object *obj) | |||
| 77 | intel_gtt_insert_sg_entries(obj->sg_list, | 100 | intel_gtt_insert_sg_entries(obj->sg_list, |
| 78 | obj->num_sg, | 101 | obj->num_sg, |
| 79 | obj->gtt_space->start >> PAGE_SHIFT, | 102 | obj->gtt_space->start >> PAGE_SHIFT, |
| 80 | obj->agp_type); | 103 | agp_type); |
| 81 | } else | 104 | } else |
| 82 | intel_gtt_insert_pages(obj->gtt_space->start >> PAGE_SHIFT, | 105 | intel_gtt_insert_pages(obj->gtt_space->start >> PAGE_SHIFT, |
| 83 | obj->base.size >> PAGE_SHIFT, | 106 | obj->base.size >> PAGE_SHIFT, |
| 84 | obj->pages, | 107 | obj->pages, |
| 85 | obj->agp_type); | 108 | agp_type); |
| 86 | 109 | ||
| 87 | return 0; | 110 | return 0; |
| 88 | } | 111 | } |
diff --git a/drivers/gpu/drm/i915/i915_gem_tiling.c b/drivers/gpu/drm/i915/i915_gem_tiling.c index 281ad3d6115d..82d70fd9e933 100644 --- a/drivers/gpu/drm/i915/i915_gem_tiling.c +++ b/drivers/gpu/drm/i915/i915_gem_tiling.c | |||
| @@ -92,7 +92,7 @@ i915_gem_detect_bit_6_swizzle(struct drm_device *dev) | |||
| 92 | uint32_t swizzle_x = I915_BIT_6_SWIZZLE_UNKNOWN; | 92 | uint32_t swizzle_x = I915_BIT_6_SWIZZLE_UNKNOWN; |
| 93 | uint32_t swizzle_y = I915_BIT_6_SWIZZLE_UNKNOWN; | 93 | uint32_t swizzle_y = I915_BIT_6_SWIZZLE_UNKNOWN; |
| 94 | 94 | ||
| 95 | if (IS_GEN5(dev) || IS_GEN6(dev)) { | 95 | if (INTEL_INFO(dev)->gen >= 5) { |
| 96 | /* On Ironlake whatever DRAM config, GPU always do | 96 | /* On Ironlake whatever DRAM config, GPU always do |
| 97 | * same swizzling setup. | 97 | * same swizzling setup. |
| 98 | */ | 98 | */ |
diff --git a/drivers/gpu/drm/i915/i915_irq.c b/drivers/gpu/drm/i915/i915_irq.c index 188b497e5076..349a03e48481 100644 --- a/drivers/gpu/drm/i915/i915_irq.c +++ b/drivers/gpu/drm/i915/i915_irq.c | |||
| @@ -367,22 +367,30 @@ static void notify_ring(struct drm_device *dev, | |||
| 367 | jiffies + msecs_to_jiffies(DRM_I915_HANGCHECK_PERIOD)); | 367 | jiffies + msecs_to_jiffies(DRM_I915_HANGCHECK_PERIOD)); |
| 368 | } | 368 | } |
| 369 | 369 | ||
| 370 | static void gen6_pm_irq_handler(struct drm_device *dev) | 370 | static void gen6_pm_rps_work(struct work_struct *work) |
| 371 | { | 371 | { |
| 372 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | 372 | drm_i915_private_t *dev_priv = container_of(work, drm_i915_private_t, |
| 373 | rps_work); | ||
| 373 | u8 new_delay = dev_priv->cur_delay; | 374 | u8 new_delay = dev_priv->cur_delay; |
| 374 | u32 pm_iir; | 375 | u32 pm_iir, pm_imr; |
| 376 | |||
| 377 | spin_lock_irq(&dev_priv->rps_lock); | ||
| 378 | pm_iir = dev_priv->pm_iir; | ||
| 379 | dev_priv->pm_iir = 0; | ||
| 380 | pm_imr = I915_READ(GEN6_PMIMR); | ||
| 381 | spin_unlock_irq(&dev_priv->rps_lock); | ||
| 375 | 382 | ||
| 376 | pm_iir = I915_READ(GEN6_PMIIR); | ||
| 377 | if (!pm_iir) | 383 | if (!pm_iir) |
| 378 | return; | 384 | return; |
| 379 | 385 | ||
| 386 | mutex_lock(&dev_priv->dev->struct_mutex); | ||
| 380 | if (pm_iir & GEN6_PM_RP_UP_THRESHOLD) { | 387 | if (pm_iir & GEN6_PM_RP_UP_THRESHOLD) { |
| 381 | if (dev_priv->cur_delay != dev_priv->max_delay) | 388 | if (dev_priv->cur_delay != dev_priv->max_delay) |
| 382 | new_delay = dev_priv->cur_delay + 1; | 389 | new_delay = dev_priv->cur_delay + 1; |
| 383 | if (new_delay > dev_priv->max_delay) | 390 | if (new_delay > dev_priv->max_delay) |
| 384 | new_delay = dev_priv->max_delay; | 391 | new_delay = dev_priv->max_delay; |
| 385 | } else if (pm_iir & (GEN6_PM_RP_DOWN_THRESHOLD | GEN6_PM_RP_DOWN_TIMEOUT)) { | 392 | } else if (pm_iir & (GEN6_PM_RP_DOWN_THRESHOLD | GEN6_PM_RP_DOWN_TIMEOUT)) { |
| 393 | gen6_gt_force_wake_get(dev_priv); | ||
| 386 | if (dev_priv->cur_delay != dev_priv->min_delay) | 394 | if (dev_priv->cur_delay != dev_priv->min_delay) |
| 387 | new_delay = dev_priv->cur_delay - 1; | 395 | new_delay = dev_priv->cur_delay - 1; |
| 388 | if (new_delay < dev_priv->min_delay) { | 396 | if (new_delay < dev_priv->min_delay) { |
| @@ -396,13 +404,19 @@ static void gen6_pm_irq_handler(struct drm_device *dev) | |||
| 396 | I915_WRITE(GEN6_RP_INTERRUPT_LIMITS, | 404 | I915_WRITE(GEN6_RP_INTERRUPT_LIMITS, |
| 397 | I915_READ(GEN6_RP_INTERRUPT_LIMITS) & ~0x3f0000); | 405 | I915_READ(GEN6_RP_INTERRUPT_LIMITS) & ~0x3f0000); |
| 398 | } | 406 | } |
| 399 | 407 | gen6_gt_force_wake_put(dev_priv); | |
| 400 | } | 408 | } |
| 401 | 409 | ||
| 402 | gen6_set_rps(dev, new_delay); | 410 | gen6_set_rps(dev_priv->dev, new_delay); |
| 403 | dev_priv->cur_delay = new_delay; | 411 | dev_priv->cur_delay = new_delay; |
| 404 | 412 | ||
| 405 | I915_WRITE(GEN6_PMIIR, pm_iir); | 413 | /* |
| 414 | * rps_lock not held here because clearing is non-destructive. There is | ||
| 415 | * an *extremely* unlikely race with gen6_rps_enable() that is prevented | ||
| 416 | * by holding struct_mutex for the duration of the write. | ||
| 417 | */ | ||
| 418 | I915_WRITE(GEN6_PMIMR, pm_imr & ~pm_iir); | ||
| 419 | mutex_unlock(&dev_priv->dev->struct_mutex); | ||
| 406 | } | 420 | } |
| 407 | 421 | ||
| 408 | static void pch_irq_handler(struct drm_device *dev) | 422 | static void pch_irq_handler(struct drm_device *dev) |
| @@ -448,8 +462,97 @@ static void pch_irq_handler(struct drm_device *dev) | |||
| 448 | DRM_DEBUG_DRIVER("PCH transcoder A underrun interrupt\n"); | 462 | DRM_DEBUG_DRIVER("PCH transcoder A underrun interrupt\n"); |
| 449 | } | 463 | } |
| 450 | 464 | ||
| 451 | static irqreturn_t ironlake_irq_handler(struct drm_device *dev) | 465 | irqreturn_t ivybridge_irq_handler(DRM_IRQ_ARGS) |
| 466 | { | ||
| 467 | struct drm_device *dev = (struct drm_device *) arg; | ||
| 468 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | ||
| 469 | int ret = IRQ_NONE; | ||
| 470 | u32 de_iir, gt_iir, de_ier, pch_iir, pm_iir; | ||
| 471 | struct drm_i915_master_private *master_priv; | ||
| 472 | |||
| 473 | atomic_inc(&dev_priv->irq_received); | ||
| 474 | |||
| 475 | /* disable master interrupt before clearing iir */ | ||
| 476 | de_ier = I915_READ(DEIER); | ||
| 477 | I915_WRITE(DEIER, de_ier & ~DE_MASTER_IRQ_CONTROL); | ||
| 478 | POSTING_READ(DEIER); | ||
| 479 | |||
| 480 | de_iir = I915_READ(DEIIR); | ||
| 481 | gt_iir = I915_READ(GTIIR); | ||
| 482 | pch_iir = I915_READ(SDEIIR); | ||
| 483 | pm_iir = I915_READ(GEN6_PMIIR); | ||
| 484 | |||
| 485 | if (de_iir == 0 && gt_iir == 0 && pch_iir == 0 && pm_iir == 0) | ||
| 486 | goto done; | ||
| 487 | |||
| 488 | ret = IRQ_HANDLED; | ||
| 489 | |||
| 490 | if (dev->primary->master) { | ||
| 491 | master_priv = dev->primary->master->driver_priv; | ||
| 492 | if (master_priv->sarea_priv) | ||
| 493 | master_priv->sarea_priv->last_dispatch = | ||
| 494 | READ_BREADCRUMB(dev_priv); | ||
| 495 | } | ||
| 496 | |||
| 497 | if (gt_iir & (GT_USER_INTERRUPT | GT_PIPE_NOTIFY)) | ||
| 498 | notify_ring(dev, &dev_priv->ring[RCS]); | ||
| 499 | if (gt_iir & GT_GEN6_BSD_USER_INTERRUPT) | ||
| 500 | notify_ring(dev, &dev_priv->ring[VCS]); | ||
| 501 | if (gt_iir & GT_BLT_USER_INTERRUPT) | ||
| 502 | notify_ring(dev, &dev_priv->ring[BCS]); | ||
| 503 | |||
| 504 | if (de_iir & DE_GSE_IVB) | ||
| 505 | intel_opregion_gse_intr(dev); | ||
| 506 | |||
| 507 | if (de_iir & DE_PLANEA_FLIP_DONE_IVB) { | ||
| 508 | intel_prepare_page_flip(dev, 0); | ||
| 509 | intel_finish_page_flip_plane(dev, 0); | ||
| 510 | } | ||
| 511 | |||
| 512 | if (de_iir & DE_PLANEB_FLIP_DONE_IVB) { | ||
| 513 | intel_prepare_page_flip(dev, 1); | ||
| 514 | intel_finish_page_flip_plane(dev, 1); | ||
| 515 | } | ||
| 516 | |||
| 517 | if (de_iir & DE_PIPEA_VBLANK_IVB) | ||
| 518 | drm_handle_vblank(dev, 0); | ||
| 519 | |||
| 520 | if (de_iir & DE_PIPEB_VBLANK_IVB); | ||
| 521 | drm_handle_vblank(dev, 1); | ||
| 522 | |||
| 523 | /* check event from PCH */ | ||
| 524 | if (de_iir & DE_PCH_EVENT_IVB) { | ||
| 525 | if (pch_iir & SDE_HOTPLUG_MASK_CPT) | ||
| 526 | queue_work(dev_priv->wq, &dev_priv->hotplug_work); | ||
| 527 | pch_irq_handler(dev); | ||
| 528 | } | ||
| 529 | |||
| 530 | if (pm_iir & GEN6_PM_DEFERRED_EVENTS) { | ||
| 531 | unsigned long flags; | ||
| 532 | spin_lock_irqsave(&dev_priv->rps_lock, flags); | ||
| 533 | WARN(dev_priv->pm_iir & pm_iir, "Missed a PM interrupt\n"); | ||
| 534 | I915_WRITE(GEN6_PMIMR, pm_iir); | ||
| 535 | dev_priv->pm_iir |= pm_iir; | ||
| 536 | spin_unlock_irqrestore(&dev_priv->rps_lock, flags); | ||
| 537 | queue_work(dev_priv->wq, &dev_priv->rps_work); | ||
| 538 | } | ||
| 539 | |||
| 540 | /* should clear PCH hotplug event before clear CPU irq */ | ||
| 541 | I915_WRITE(SDEIIR, pch_iir); | ||
| 542 | I915_WRITE(GTIIR, gt_iir); | ||
| 543 | I915_WRITE(DEIIR, de_iir); | ||
| 544 | I915_WRITE(GEN6_PMIIR, pm_iir); | ||
| 545 | |||
| 546 | done: | ||
| 547 | I915_WRITE(DEIER, de_ier); | ||
| 548 | POSTING_READ(DEIER); | ||
| 549 | |||
| 550 | return ret; | ||
| 551 | } | ||
| 552 | |||
| 553 | irqreturn_t ironlake_irq_handler(DRM_IRQ_ARGS) | ||
| 452 | { | 554 | { |
| 555 | struct drm_device *dev = (struct drm_device *) arg; | ||
| 453 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | 556 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; |
| 454 | int ret = IRQ_NONE; | 557 | int ret = IRQ_NONE; |
| 455 | u32 de_iir, gt_iir, de_ier, pch_iir, pm_iir; | 558 | u32 de_iir, gt_iir, de_ier, pch_iir, pm_iir; |
| @@ -457,6 +560,8 @@ static irqreturn_t ironlake_irq_handler(struct drm_device *dev) | |||
| 457 | struct drm_i915_master_private *master_priv; | 560 | struct drm_i915_master_private *master_priv; |
| 458 | u32 bsd_usr_interrupt = GT_BSD_USER_INTERRUPT; | 561 | u32 bsd_usr_interrupt = GT_BSD_USER_INTERRUPT; |
| 459 | 562 | ||
| 563 | atomic_inc(&dev_priv->irq_received); | ||
| 564 | |||
| 460 | if (IS_GEN6(dev)) | 565 | if (IS_GEN6(dev)) |
| 461 | bsd_usr_interrupt = GT_GEN6_BSD_USER_INTERRUPT; | 566 | bsd_usr_interrupt = GT_GEN6_BSD_USER_INTERRUPT; |
| 462 | 567 | ||
| @@ -526,13 +631,30 @@ static irqreturn_t ironlake_irq_handler(struct drm_device *dev) | |||
| 526 | i915_handle_rps_change(dev); | 631 | i915_handle_rps_change(dev); |
| 527 | } | 632 | } |
| 528 | 633 | ||
| 529 | if (IS_GEN6(dev)) | 634 | if (IS_GEN6(dev) && pm_iir & GEN6_PM_DEFERRED_EVENTS) { |
| 530 | gen6_pm_irq_handler(dev); | 635 | /* |
| 636 | * IIR bits should never already be set because IMR should | ||
| 637 | * prevent an interrupt from being shown in IIR. The warning | ||
| 638 | * displays a case where we've unsafely cleared | ||
| 639 | * dev_priv->pm_iir. Although missing an interrupt of the same | ||
| 640 | * type is not a problem, it displays a problem in the logic. | ||
| 641 | * | ||
| 642 | * The mask bit in IMR is cleared by rps_work. | ||
| 643 | */ | ||
| 644 | unsigned long flags; | ||
| 645 | spin_lock_irqsave(&dev_priv->rps_lock, flags); | ||
| 646 | WARN(dev_priv->pm_iir & pm_iir, "Missed a PM interrupt\n"); | ||
| 647 | I915_WRITE(GEN6_PMIMR, pm_iir); | ||
| 648 | dev_priv->pm_iir |= pm_iir; | ||
| 649 | spin_unlock_irqrestore(&dev_priv->rps_lock, flags); | ||
| 650 | queue_work(dev_priv->wq, &dev_priv->rps_work); | ||
| 651 | } | ||
| 531 | 652 | ||
| 532 | /* should clear PCH hotplug event before clear CPU irq */ | 653 | /* should clear PCH hotplug event before clear CPU irq */ |
| 533 | I915_WRITE(SDEIIR, pch_iir); | 654 | I915_WRITE(SDEIIR, pch_iir); |
| 534 | I915_WRITE(GTIIR, gt_iir); | 655 | I915_WRITE(GTIIR, gt_iir); |
| 535 | I915_WRITE(DEIIR, de_iir); | 656 | I915_WRITE(DEIIR, de_iir); |
| 657 | I915_WRITE(GEN6_PMIIR, pm_iir); | ||
| 536 | 658 | ||
| 537 | done: | 659 | done: |
| 538 | I915_WRITE(DEIER, de_ier); | 660 | I915_WRITE(DEIER, de_ier); |
| @@ -676,7 +798,7 @@ static u32 capture_bo_list(struct drm_i915_error_buffer *err, | |||
| 676 | err->dirty = obj->dirty; | 798 | err->dirty = obj->dirty; |
| 677 | err->purgeable = obj->madv != I915_MADV_WILLNEED; | 799 | err->purgeable = obj->madv != I915_MADV_WILLNEED; |
| 678 | err->ring = obj->ring ? obj->ring->id : 0; | 800 | err->ring = obj->ring ? obj->ring->id : 0; |
| 679 | err->agp_type = obj->agp_type == AGP_USER_CACHED_MEMORY; | 801 | err->cache_level = obj->cache_level; |
| 680 | 802 | ||
| 681 | if (++i == count) | 803 | if (++i == count) |
| 682 | break; | 804 | break; |
| @@ -1103,9 +1225,6 @@ irqreturn_t i915_driver_irq_handler(DRM_IRQ_ARGS) | |||
| 1103 | 1225 | ||
| 1104 | atomic_inc(&dev_priv->irq_received); | 1226 | atomic_inc(&dev_priv->irq_received); |
| 1105 | 1227 | ||
| 1106 | if (HAS_PCH_SPLIT(dev)) | ||
| 1107 | return ironlake_irq_handler(dev); | ||
| 1108 | |||
| 1109 | iir = I915_READ(IIR); | 1228 | iir = I915_READ(IIR); |
| 1110 | 1229 | ||
| 1111 | if (INTEL_INFO(dev)->gen >= 4) | 1230 | if (INTEL_INFO(dev)->gen >= 4) |
| @@ -1344,10 +1463,7 @@ int i915_enable_vblank(struct drm_device *dev, int pipe) | |||
| 1344 | return -EINVAL; | 1463 | return -EINVAL; |
| 1345 | 1464 | ||
| 1346 | spin_lock_irqsave(&dev_priv->irq_lock, irqflags); | 1465 | spin_lock_irqsave(&dev_priv->irq_lock, irqflags); |
| 1347 | if (HAS_PCH_SPLIT(dev)) | 1466 | if (INTEL_INFO(dev)->gen >= 4) |
| 1348 | ironlake_enable_display_irq(dev_priv, (pipe == 0) ? | ||
| 1349 | DE_PIPEA_VBLANK: DE_PIPEB_VBLANK); | ||
| 1350 | else if (INTEL_INFO(dev)->gen >= 4) | ||
| 1351 | i915_enable_pipestat(dev_priv, pipe, | 1467 | i915_enable_pipestat(dev_priv, pipe, |
| 1352 | PIPE_START_VBLANK_INTERRUPT_ENABLE); | 1468 | PIPE_START_VBLANK_INTERRUPT_ENABLE); |
| 1353 | else | 1469 | else |
| @@ -1362,6 +1478,38 @@ int i915_enable_vblank(struct drm_device *dev, int pipe) | |||
| 1362 | return 0; | 1478 | return 0; |
| 1363 | } | 1479 | } |
| 1364 | 1480 | ||
| 1481 | int ironlake_enable_vblank(struct drm_device *dev, int pipe) | ||
| 1482 | { | ||
| 1483 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | ||
| 1484 | unsigned long irqflags; | ||
| 1485 | |||
| 1486 | if (!i915_pipe_enabled(dev, pipe)) | ||
| 1487 | return -EINVAL; | ||
| 1488 | |||
| 1489 | spin_lock_irqsave(&dev_priv->irq_lock, irqflags); | ||
| 1490 | ironlake_enable_display_irq(dev_priv, (pipe == 0) ? | ||
| 1491 | DE_PIPEA_VBLANK: DE_PIPEB_VBLANK); | ||
| 1492 | spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); | ||
| 1493 | |||
| 1494 | return 0; | ||
| 1495 | } | ||
| 1496 | |||
| 1497 | int ivybridge_enable_vblank(struct drm_device *dev, int pipe) | ||
| 1498 | { | ||
| 1499 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | ||
| 1500 | unsigned long irqflags; | ||
| 1501 | |||
| 1502 | if (!i915_pipe_enabled(dev, pipe)) | ||
| 1503 | return -EINVAL; | ||
| 1504 | |||
| 1505 | spin_lock_irqsave(&dev_priv->irq_lock, irqflags); | ||
| 1506 | ironlake_enable_display_irq(dev_priv, (pipe == 0) ? | ||
| 1507 | DE_PIPEA_VBLANK_IVB : DE_PIPEB_VBLANK_IVB); | ||
| 1508 | spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); | ||
| 1509 | |||
| 1510 | return 0; | ||
| 1511 | } | ||
| 1512 | |||
| 1365 | /* Called from drm generic code, passed 'crtc' which | 1513 | /* Called from drm generic code, passed 'crtc' which |
| 1366 | * we use as a pipe index | 1514 | * we use as a pipe index |
| 1367 | */ | 1515 | */ |
| @@ -1375,13 +1523,31 @@ void i915_disable_vblank(struct drm_device *dev, int pipe) | |||
| 1375 | I915_WRITE(INSTPM, | 1523 | I915_WRITE(INSTPM, |
| 1376 | INSTPM_AGPBUSY_DIS << 16 | INSTPM_AGPBUSY_DIS); | 1524 | INSTPM_AGPBUSY_DIS << 16 | INSTPM_AGPBUSY_DIS); |
| 1377 | 1525 | ||
| 1378 | if (HAS_PCH_SPLIT(dev)) | 1526 | i915_disable_pipestat(dev_priv, pipe, |
| 1379 | ironlake_disable_display_irq(dev_priv, (pipe == 0) ? | 1527 | PIPE_VBLANK_INTERRUPT_ENABLE | |
| 1380 | DE_PIPEA_VBLANK: DE_PIPEB_VBLANK); | 1528 | PIPE_START_VBLANK_INTERRUPT_ENABLE); |
| 1381 | else | 1529 | spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); |
| 1382 | i915_disable_pipestat(dev_priv, pipe, | 1530 | } |
| 1383 | PIPE_VBLANK_INTERRUPT_ENABLE | | 1531 | |
| 1384 | PIPE_START_VBLANK_INTERRUPT_ENABLE); | 1532 | void ironlake_disable_vblank(struct drm_device *dev, int pipe) |
| 1533 | { | ||
| 1534 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | ||
| 1535 | unsigned long irqflags; | ||
| 1536 | |||
| 1537 | spin_lock_irqsave(&dev_priv->irq_lock, irqflags); | ||
| 1538 | ironlake_disable_display_irq(dev_priv, (pipe == 0) ? | ||
| 1539 | DE_PIPEA_VBLANK: DE_PIPEB_VBLANK); | ||
| 1540 | spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); | ||
| 1541 | } | ||
| 1542 | |||
| 1543 | void ivybridge_disable_vblank(struct drm_device *dev, int pipe) | ||
| 1544 | { | ||
| 1545 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | ||
| 1546 | unsigned long irqflags; | ||
| 1547 | |||
| 1548 | spin_lock_irqsave(&dev_priv->irq_lock, irqflags); | ||
| 1549 | ironlake_disable_display_irq(dev_priv, (pipe == 0) ? | ||
| 1550 | DE_PIPEA_VBLANK_IVB : DE_PIPEB_VBLANK_IVB); | ||
| 1385 | spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); | 1551 | spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags); |
| 1386 | } | 1552 | } |
| 1387 | 1553 | ||
| @@ -1562,10 +1728,15 @@ repeat: | |||
| 1562 | 1728 | ||
| 1563 | /* drm_dma.h hooks | 1729 | /* drm_dma.h hooks |
| 1564 | */ | 1730 | */ |
| 1565 | static void ironlake_irq_preinstall(struct drm_device *dev) | 1731 | void ironlake_irq_preinstall(struct drm_device *dev) |
| 1566 | { | 1732 | { |
| 1567 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | 1733 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; |
| 1568 | 1734 | ||
| 1735 | atomic_set(&dev_priv->irq_received, 0); | ||
| 1736 | |||
| 1737 | INIT_WORK(&dev_priv->hotplug_work, i915_hotplug_work_func); | ||
| 1738 | INIT_WORK(&dev_priv->error_work, i915_error_work_func); | ||
| 1739 | |||
| 1569 | I915_WRITE(HWSTAM, 0xeffe); | 1740 | I915_WRITE(HWSTAM, 0xeffe); |
| 1570 | 1741 | ||
| 1571 | /* XXX hotplug from PCH */ | 1742 | /* XXX hotplug from PCH */ |
| @@ -1585,7 +1756,7 @@ static void ironlake_irq_preinstall(struct drm_device *dev) | |||
| 1585 | POSTING_READ(SDEIER); | 1756 | POSTING_READ(SDEIER); |
| 1586 | } | 1757 | } |
| 1587 | 1758 | ||
| 1588 | static int ironlake_irq_postinstall(struct drm_device *dev) | 1759 | int ironlake_irq_postinstall(struct drm_device *dev) |
| 1589 | { | 1760 | { |
| 1590 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | 1761 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; |
| 1591 | /* enable kind of interrupts always enabled */ | 1762 | /* enable kind of interrupts always enabled */ |
| @@ -1594,6 +1765,13 @@ static int ironlake_irq_postinstall(struct drm_device *dev) | |||
| 1594 | u32 render_irqs; | 1765 | u32 render_irqs; |
| 1595 | u32 hotplug_mask; | 1766 | u32 hotplug_mask; |
| 1596 | 1767 | ||
| 1768 | DRM_INIT_WAITQUEUE(&dev_priv->ring[RCS].irq_queue); | ||
| 1769 | if (HAS_BSD(dev)) | ||
| 1770 | DRM_INIT_WAITQUEUE(&dev_priv->ring[VCS].irq_queue); | ||
| 1771 | if (HAS_BLT(dev)) | ||
| 1772 | DRM_INIT_WAITQUEUE(&dev_priv->ring[BCS].irq_queue); | ||
| 1773 | |||
| 1774 | dev_priv->vblank_pipe = DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B; | ||
| 1597 | dev_priv->irq_mask = ~display_mask; | 1775 | dev_priv->irq_mask = ~display_mask; |
| 1598 | 1776 | ||
| 1599 | /* should always can generate irq */ | 1777 | /* should always can generate irq */ |
| @@ -1650,6 +1828,56 @@ static int ironlake_irq_postinstall(struct drm_device *dev) | |||
| 1650 | return 0; | 1828 | return 0; |
| 1651 | } | 1829 | } |
| 1652 | 1830 | ||
| 1831 | int ivybridge_irq_postinstall(struct drm_device *dev) | ||
| 1832 | { | ||
| 1833 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | ||
| 1834 | /* enable kind of interrupts always enabled */ | ||
| 1835 | u32 display_mask = DE_MASTER_IRQ_CONTROL | DE_GSE_IVB | | ||
| 1836 | DE_PCH_EVENT_IVB | DE_PLANEA_FLIP_DONE_IVB | | ||
| 1837 | DE_PLANEB_FLIP_DONE_IVB; | ||
| 1838 | u32 render_irqs; | ||
| 1839 | u32 hotplug_mask; | ||
| 1840 | |||
| 1841 | DRM_INIT_WAITQUEUE(&dev_priv->ring[RCS].irq_queue); | ||
| 1842 | if (HAS_BSD(dev)) | ||
| 1843 | DRM_INIT_WAITQUEUE(&dev_priv->ring[VCS].irq_queue); | ||
| 1844 | if (HAS_BLT(dev)) | ||
| 1845 | DRM_INIT_WAITQUEUE(&dev_priv->ring[BCS].irq_queue); | ||
| 1846 | |||
| 1847 | dev_priv->vblank_pipe = DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B; | ||
| 1848 | dev_priv->irq_mask = ~display_mask; | ||
| 1849 | |||
| 1850 | /* should always can generate irq */ | ||
| 1851 | I915_WRITE(DEIIR, I915_READ(DEIIR)); | ||
| 1852 | I915_WRITE(DEIMR, dev_priv->irq_mask); | ||
| 1853 | I915_WRITE(DEIER, display_mask | DE_PIPEA_VBLANK_IVB | | ||
| 1854 | DE_PIPEB_VBLANK_IVB); | ||
| 1855 | POSTING_READ(DEIER); | ||
| 1856 | |||
| 1857 | dev_priv->gt_irq_mask = ~0; | ||
| 1858 | |||
| 1859 | I915_WRITE(GTIIR, I915_READ(GTIIR)); | ||
| 1860 | I915_WRITE(GTIMR, dev_priv->gt_irq_mask); | ||
| 1861 | |||
| 1862 | render_irqs = GT_USER_INTERRUPT | GT_GEN6_BSD_USER_INTERRUPT | | ||
| 1863 | GT_BLT_USER_INTERRUPT; | ||
| 1864 | I915_WRITE(GTIER, render_irqs); | ||
| 1865 | POSTING_READ(GTIER); | ||
| 1866 | |||
| 1867 | hotplug_mask = (SDE_CRT_HOTPLUG_CPT | | ||
| 1868 | SDE_PORTB_HOTPLUG_CPT | | ||
| 1869 | SDE_PORTC_HOTPLUG_CPT | | ||
| 1870 | SDE_PORTD_HOTPLUG_CPT); | ||
| 1871 | dev_priv->pch_irq_mask = ~hotplug_mask; | ||
| 1872 | |||
| 1873 | I915_WRITE(SDEIIR, I915_READ(SDEIIR)); | ||
| 1874 | I915_WRITE(SDEIMR, dev_priv->pch_irq_mask); | ||
| 1875 | I915_WRITE(SDEIER, hotplug_mask); | ||
| 1876 | POSTING_READ(SDEIER); | ||
| 1877 | |||
| 1878 | return 0; | ||
| 1879 | } | ||
| 1880 | |||
| 1653 | void i915_driver_irq_preinstall(struct drm_device * dev) | 1881 | void i915_driver_irq_preinstall(struct drm_device * dev) |
| 1654 | { | 1882 | { |
| 1655 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | 1883 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; |
| @@ -1659,11 +1887,7 @@ void i915_driver_irq_preinstall(struct drm_device * dev) | |||
| 1659 | 1887 | ||
| 1660 | INIT_WORK(&dev_priv->hotplug_work, i915_hotplug_work_func); | 1888 | INIT_WORK(&dev_priv->hotplug_work, i915_hotplug_work_func); |
| 1661 | INIT_WORK(&dev_priv->error_work, i915_error_work_func); | 1889 | INIT_WORK(&dev_priv->error_work, i915_error_work_func); |
| 1662 | 1890 | INIT_WORK(&dev_priv->rps_work, gen6_pm_rps_work); | |
| 1663 | if (HAS_PCH_SPLIT(dev)) { | ||
| 1664 | ironlake_irq_preinstall(dev); | ||
| 1665 | return; | ||
| 1666 | } | ||
| 1667 | 1891 | ||
| 1668 | if (I915_HAS_HOTPLUG(dev)) { | 1892 | if (I915_HAS_HOTPLUG(dev)) { |
| 1669 | I915_WRITE(PORT_HOTPLUG_EN, 0); | 1893 | I915_WRITE(PORT_HOTPLUG_EN, 0); |
| @@ -1688,17 +1912,8 @@ int i915_driver_irq_postinstall(struct drm_device *dev) | |||
| 1688 | u32 enable_mask = I915_INTERRUPT_ENABLE_FIX | I915_INTERRUPT_ENABLE_VAR; | 1912 | u32 enable_mask = I915_INTERRUPT_ENABLE_FIX | I915_INTERRUPT_ENABLE_VAR; |
| 1689 | u32 error_mask; | 1913 | u32 error_mask; |
| 1690 | 1914 | ||
| 1691 | DRM_INIT_WAITQUEUE(&dev_priv->ring[RCS].irq_queue); | ||
| 1692 | if (HAS_BSD(dev)) | ||
| 1693 | DRM_INIT_WAITQUEUE(&dev_priv->ring[VCS].irq_queue); | ||
| 1694 | if (HAS_BLT(dev)) | ||
| 1695 | DRM_INIT_WAITQUEUE(&dev_priv->ring[BCS].irq_queue); | ||
| 1696 | |||
| 1697 | dev_priv->vblank_pipe = DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B; | 1915 | dev_priv->vblank_pipe = DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B; |
| 1698 | 1916 | ||
| 1699 | if (HAS_PCH_SPLIT(dev)) | ||
| 1700 | return ironlake_irq_postinstall(dev); | ||
| 1701 | |||
| 1702 | /* Unmask the interrupts that we always want on. */ | 1917 | /* Unmask the interrupts that we always want on. */ |
| 1703 | dev_priv->irq_mask = ~I915_INTERRUPT_ENABLE_FIX; | 1918 | dev_priv->irq_mask = ~I915_INTERRUPT_ENABLE_FIX; |
| 1704 | 1919 | ||
| @@ -1767,9 +1982,15 @@ int i915_driver_irq_postinstall(struct drm_device *dev) | |||
| 1767 | return 0; | 1982 | return 0; |
| 1768 | } | 1983 | } |
| 1769 | 1984 | ||
| 1770 | static void ironlake_irq_uninstall(struct drm_device *dev) | 1985 | void ironlake_irq_uninstall(struct drm_device *dev) |
| 1771 | { | 1986 | { |
| 1772 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; | 1987 | drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private; |
| 1988 | |||
| 1989 | if (!dev_priv) | ||
| 1990 | return; | ||
| 1991 | |||
| 1992 | dev_priv->vblank_pipe = 0; | ||
| 1993 | |||
| 1773 | I915_WRITE(HWSTAM, 0xffffffff); | 1994 | I915_WRITE(HWSTAM, 0xffffffff); |
| 1774 | 1995 | ||
| 1775 | I915_WRITE(DEIMR, 0xffffffff); | 1996 | I915_WRITE(DEIMR, 0xffffffff); |
| @@ -1791,11 +2012,6 @@ void i915_driver_irq_uninstall(struct drm_device * dev) | |||
| 1791 | 2012 | ||
| 1792 | dev_priv->vblank_pipe = 0; | 2013 | dev_priv->vblank_pipe = 0; |
| 1793 | 2014 | ||
| 1794 | if (HAS_PCH_SPLIT(dev)) { | ||
| 1795 | ironlake_irq_uninstall(dev); | ||
| 1796 | return; | ||
| 1797 | } | ||
| 1798 | |||
| 1799 | if (I915_HAS_HOTPLUG(dev)) { | 2015 | if (I915_HAS_HOTPLUG(dev)) { |
| 1800 | I915_WRITE(PORT_HOTPLUG_EN, 0); | 2016 | I915_WRITE(PORT_HOTPLUG_EN, 0); |
| 1801 | I915_WRITE(PORT_HOTPLUG_STAT, I915_READ(PORT_HOTPLUG_STAT)); | 2017 | I915_WRITE(PORT_HOTPLUG_STAT, I915_READ(PORT_HOTPLUG_STAT)); |
diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h index f39ac3a0fa93..2f967af8e62e 100644 --- a/drivers/gpu/drm/i915/i915_reg.h +++ b/drivers/gpu/drm/i915/i915_reg.h | |||
| @@ -291,6 +291,9 @@ | |||
| 291 | #define RING_MAX_IDLE(base) ((base)+0x54) | 291 | #define RING_MAX_IDLE(base) ((base)+0x54) |
| 292 | #define RING_HWS_PGA(base) ((base)+0x80) | 292 | #define RING_HWS_PGA(base) ((base)+0x80) |
| 293 | #define RING_HWS_PGA_GEN6(base) ((base)+0x2080) | 293 | #define RING_HWS_PGA_GEN6(base) ((base)+0x2080) |
| 294 | #define RENDER_HWS_PGA_GEN7 (0x04080) | ||
| 295 | #define BSD_HWS_PGA_GEN7 (0x04180) | ||
| 296 | #define BLT_HWS_PGA_GEN7 (0x04280) | ||
| 294 | #define RING_ACTHD(base) ((base)+0x74) | 297 | #define RING_ACTHD(base) ((base)+0x74) |
| 295 | #define RING_NOPID(base) ((base)+0x94) | 298 | #define RING_NOPID(base) ((base)+0x94) |
| 296 | #define RING_IMR(base) ((base)+0xa8) | 299 | #define RING_IMR(base) ((base)+0xa8) |
| @@ -2778,6 +2781,19 @@ | |||
| 2778 | #define DE_PIPEA_VSYNC (1 << 3) | 2781 | #define DE_PIPEA_VSYNC (1 << 3) |
| 2779 | #define DE_PIPEA_FIFO_UNDERRUN (1 << 0) | 2782 | #define DE_PIPEA_FIFO_UNDERRUN (1 << 0) |
| 2780 | 2783 | ||
| 2784 | /* More Ivybridge lolz */ | ||
| 2785 | #define DE_ERR_DEBUG_IVB (1<<30) | ||
| 2786 | #define DE_GSE_IVB (1<<29) | ||
| 2787 | #define DE_PCH_EVENT_IVB (1<<28) | ||
| 2788 | #define DE_DP_A_HOTPLUG_IVB (1<<27) | ||
| 2789 | #define DE_AUX_CHANNEL_A_IVB (1<<26) | ||
| 2790 | #define DE_SPRITEB_FLIP_DONE_IVB (1<<9) | ||
| 2791 | #define DE_SPRITEA_FLIP_DONE_IVB (1<<4) | ||
| 2792 | #define DE_PLANEB_FLIP_DONE_IVB (1<<8) | ||
| 2793 | #define DE_PLANEA_FLIP_DONE_IVB (1<<3) | ||
| 2794 | #define DE_PIPEB_VBLANK_IVB (1<<5) | ||
| 2795 | #define DE_PIPEA_VBLANK_IVB (1<<0) | ||
| 2796 | |||
| 2781 | #define DEISR 0x44000 | 2797 | #define DEISR 0x44000 |
| 2782 | #define DEIMR 0x44004 | 2798 | #define DEIMR 0x44004 |
| 2783 | #define DEIIR 0x44008 | 2799 | #define DEIIR 0x44008 |
| @@ -2809,6 +2825,7 @@ | |||
| 2809 | #define ILK_eDP_A_DISABLE (1<<24) | 2825 | #define ILK_eDP_A_DISABLE (1<<24) |
| 2810 | #define ILK_DESKTOP (1<<23) | 2826 | #define ILK_DESKTOP (1<<23) |
| 2811 | #define ILK_DSPCLK_GATE 0x42020 | 2827 | #define ILK_DSPCLK_GATE 0x42020 |
| 2828 | #define IVB_VRHUNIT_CLK_GATE (1<<28) | ||
| 2812 | #define ILK_DPARB_CLK_GATE (1<<5) | 2829 | #define ILK_DPARB_CLK_GATE (1<<5) |
| 2813 | #define ILK_DPFD_CLK_GATE (1<<7) | 2830 | #define ILK_DPFD_CLK_GATE (1<<7) |
| 2814 | 2831 | ||
| @@ -3057,6 +3074,9 @@ | |||
| 3057 | #define TRANS_6BPC (2<<5) | 3074 | #define TRANS_6BPC (2<<5) |
| 3058 | #define TRANS_12BPC (3<<5) | 3075 | #define TRANS_12BPC (3<<5) |
| 3059 | 3076 | ||
| 3077 | #define SOUTH_CHICKEN2 0xc2004 | ||
| 3078 | #define DPLS_EDP_PPS_FIX_DIS (1<<0) | ||
| 3079 | |||
| 3060 | #define _FDI_RXA_CHICKEN 0xc200c | 3080 | #define _FDI_RXA_CHICKEN 0xc200c |
| 3061 | #define _FDI_RXB_CHICKEN 0xc2010 | 3081 | #define _FDI_RXB_CHICKEN 0xc2010 |
| 3062 | #define FDI_RX_PHASE_SYNC_POINTER_OVR (1<<1) | 3082 | #define FDI_RX_PHASE_SYNC_POINTER_OVR (1<<1) |
| @@ -3104,7 +3124,15 @@ | |||
| 3104 | #define FDI_TX_ENHANCE_FRAME_ENABLE (1<<18) | 3124 | #define FDI_TX_ENHANCE_FRAME_ENABLE (1<<18) |
| 3105 | /* Ironlake: hardwired to 1 */ | 3125 | /* Ironlake: hardwired to 1 */ |
| 3106 | #define FDI_TX_PLL_ENABLE (1<<14) | 3126 | #define FDI_TX_PLL_ENABLE (1<<14) |
| 3127 | |||
| 3128 | /* Ivybridge has different bits for lolz */ | ||
| 3129 | #define FDI_LINK_TRAIN_PATTERN_1_IVB (0<<8) | ||
| 3130 | #define FDI_LINK_TRAIN_PATTERN_2_IVB (1<<8) | ||
| 3131 | #define FDI_LINK_TRAIN_PATTERN_IDLE_IVB (2<<8) | ||
| 3132 | #define FDI_LINK_TRAIN_NONE_IVB (3<<8) | ||
| 3133 | |||
| 3107 | /* both Tx and Rx */ | 3134 | /* both Tx and Rx */ |
| 3135 | #define FDI_LINK_TRAIN_AUTO (1<<10) | ||
| 3108 | #define FDI_SCRAMBLING_ENABLE (0<<7) | 3136 | #define FDI_SCRAMBLING_ENABLE (0<<7) |
| 3109 | #define FDI_SCRAMBLING_DISABLE (1<<7) | 3137 | #define FDI_SCRAMBLING_DISABLE (1<<7) |
| 3110 | 3138 | ||
| @@ -3114,6 +3142,8 @@ | |||
| 3114 | #define FDI_RX_CTL(pipe) _PIPE(pipe, _FDI_RXA_CTL, _FDI_RXB_CTL) | 3142 | #define FDI_RX_CTL(pipe) _PIPE(pipe, _FDI_RXA_CTL, _FDI_RXB_CTL) |
| 3115 | #define FDI_RX_ENABLE (1<<31) | 3143 | #define FDI_RX_ENABLE (1<<31) |
| 3116 | /* train, dp width same as FDI_TX */ | 3144 | /* train, dp width same as FDI_TX */ |
| 3145 | #define FDI_FS_ERRC_ENABLE (1<<27) | ||
| 3146 | #define FDI_FE_ERRC_ENABLE (1<<26) | ||
| 3117 | #define FDI_DP_PORT_WIDTH_X8 (7<<19) | 3147 | #define FDI_DP_PORT_WIDTH_X8 (7<<19) |
| 3118 | #define FDI_8BPC (0<<16) | 3148 | #define FDI_8BPC (0<<16) |
| 3119 | #define FDI_10BPC (1<<16) | 3149 | #define FDI_10BPC (1<<16) |
| @@ -3386,7 +3416,7 @@ | |||
| 3386 | #define GEN6_PMINTRMSK 0xA168 | 3416 | #define GEN6_PMINTRMSK 0xA168 |
| 3387 | 3417 | ||
| 3388 | #define GEN6_PMISR 0x44020 | 3418 | #define GEN6_PMISR 0x44020 |
| 3389 | #define GEN6_PMIMR 0x44024 | 3419 | #define GEN6_PMIMR 0x44024 /* rps_lock */ |
| 3390 | #define GEN6_PMIIR 0x44028 | 3420 | #define GEN6_PMIIR 0x44028 |
| 3391 | #define GEN6_PMIER 0x4402C | 3421 | #define GEN6_PMIER 0x4402C |
| 3392 | #define GEN6_PM_MBOX_EVENT (1<<25) | 3422 | #define GEN6_PM_MBOX_EVENT (1<<25) |
| @@ -3396,6 +3426,9 @@ | |||
| 3396 | #define GEN6_PM_RP_DOWN_THRESHOLD (1<<4) | 3426 | #define GEN6_PM_RP_DOWN_THRESHOLD (1<<4) |
| 3397 | #define GEN6_PM_RP_UP_EI_EXPIRED (1<<2) | 3427 | #define GEN6_PM_RP_UP_EI_EXPIRED (1<<2) |
| 3398 | #define GEN6_PM_RP_DOWN_EI_EXPIRED (1<<1) | 3428 | #define GEN6_PM_RP_DOWN_EI_EXPIRED (1<<1) |
| 3429 | #define GEN6_PM_DEFERRED_EVENTS (GEN6_PM_RP_UP_THRESHOLD | \ | ||
| 3430 | GEN6_PM_RP_DOWN_THRESHOLD | \ | ||
| 3431 | GEN6_PM_RP_DOWN_TIMEOUT) | ||
| 3399 | 3432 | ||
| 3400 | #define GEN6_PCODE_MAILBOX 0x138124 | 3433 | #define GEN6_PCODE_MAILBOX 0x138124 |
| 3401 | #define GEN6_PCODE_READY (1<<31) | 3434 | #define GEN6_PCODE_READY (1<<31) |
diff --git a/drivers/gpu/drm/i915/i915_suspend.c b/drivers/gpu/drm/i915/i915_suspend.c index da474153a0a2..60a94d2b5264 100644 --- a/drivers/gpu/drm/i915/i915_suspend.c +++ b/drivers/gpu/drm/i915/i915_suspend.c | |||
| @@ -863,8 +863,7 @@ int i915_restore_state(struct drm_device *dev) | |||
| 863 | I915_WRITE(IMR, dev_priv->saveIMR); | 863 | I915_WRITE(IMR, dev_priv->saveIMR); |
| 864 | } | 864 | } |
| 865 | 865 | ||
| 866 | /* Clock gating state */ | 866 | intel_init_clock_gating(dev); |
| 867 | intel_enable_clock_gating(dev); | ||
| 868 | 867 | ||
| 869 | if (IS_IRONLAKE_M(dev)) { | 868 | if (IS_IRONLAKE_M(dev)) { |
| 870 | ironlake_enable_drps(dev); | 869 | ironlake_enable_drps(dev); |
diff --git a/drivers/gpu/drm/i915/intel_crt.c b/drivers/gpu/drm/i915/intel_crt.c index d03fc05b39c0..e93f93cc7e78 100644 --- a/drivers/gpu/drm/i915/intel_crt.c +++ b/drivers/gpu/drm/i915/intel_crt.c | |||
| @@ -305,13 +305,11 @@ static bool intel_crt_detect_ddc(struct drm_connector *connector) | |||
| 305 | } | 305 | } |
| 306 | 306 | ||
| 307 | static enum drm_connector_status | 307 | static enum drm_connector_status |
| 308 | intel_crt_load_detect(struct drm_crtc *crtc, struct intel_crt *crt) | 308 | intel_crt_load_detect(struct intel_crt *crt) |
| 309 | { | 309 | { |
| 310 | struct drm_encoder *encoder = &crt->base.base; | 310 | struct drm_device *dev = crt->base.base.dev; |
| 311 | struct drm_device *dev = encoder->dev; | ||
| 312 | struct drm_i915_private *dev_priv = dev->dev_private; | 311 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 313 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); | 312 | uint32_t pipe = to_intel_crtc(crt->base.base.crtc)->pipe; |
| 314 | uint32_t pipe = intel_crtc->pipe; | ||
| 315 | uint32_t save_bclrpat; | 313 | uint32_t save_bclrpat; |
| 316 | uint32_t save_vtotal; | 314 | uint32_t save_vtotal; |
| 317 | uint32_t vtotal, vactive; | 315 | uint32_t vtotal, vactive; |
| @@ -432,7 +430,6 @@ intel_crt_detect(struct drm_connector *connector, bool force) | |||
| 432 | struct drm_device *dev = connector->dev; | 430 | struct drm_device *dev = connector->dev; |
| 433 | struct intel_crt *crt = intel_attached_crt(connector); | 431 | struct intel_crt *crt = intel_attached_crt(connector); |
| 434 | struct drm_crtc *crtc; | 432 | struct drm_crtc *crtc; |
| 435 | int dpms_mode; | ||
| 436 | enum drm_connector_status status; | 433 | enum drm_connector_status status; |
| 437 | 434 | ||
| 438 | if (I915_HAS_HOTPLUG(dev)) { | 435 | if (I915_HAS_HOTPLUG(dev)) { |
| @@ -454,17 +451,18 @@ intel_crt_detect(struct drm_connector *connector, bool force) | |||
| 454 | /* for pre-945g platforms use load detect */ | 451 | /* for pre-945g platforms use load detect */ |
| 455 | crtc = crt->base.base.crtc; | 452 | crtc = crt->base.base.crtc; |
| 456 | if (crtc && crtc->enabled) { | 453 | if (crtc && crtc->enabled) { |
| 457 | status = intel_crt_load_detect(crtc, crt); | 454 | status = intel_crt_load_detect(crt); |
| 458 | } else { | 455 | } else { |
| 459 | crtc = intel_get_load_detect_pipe(&crt->base, connector, | 456 | struct intel_load_detect_pipe tmp; |
| 460 | NULL, &dpms_mode); | 457 | |
| 461 | if (crtc) { | 458 | if (intel_get_load_detect_pipe(&crt->base, connector, NULL, |
| 459 | &tmp)) { | ||
| 462 | if (intel_crt_detect_ddc(connector)) | 460 | if (intel_crt_detect_ddc(connector)) |
| 463 | status = connector_status_connected; | 461 | status = connector_status_connected; |
| 464 | else | 462 | else |
| 465 | status = intel_crt_load_detect(crtc, crt); | 463 | status = intel_crt_load_detect(crt); |
| 466 | intel_release_load_detect_pipe(&crt->base, | 464 | intel_release_load_detect_pipe(&crt->base, connector, |
| 467 | connector, dpms_mode); | 465 | &tmp); |
| 468 | } else | 466 | } else |
| 469 | status = connector_status_unknown; | 467 | status = connector_status_unknown; |
| 470 | } | 468 | } |
diff --git a/drivers/gpu/drm/i915/intel_display.c b/drivers/gpu/drm/i915/intel_display.c index 21a7e70feacc..565eb2cc0042 100644 --- a/drivers/gpu/drm/i915/intel_display.c +++ b/drivers/gpu/drm/i915/intel_display.c | |||
| @@ -76,255 +76,6 @@ struct intel_limit { | |||
| 76 | int, int, intel_clock_t *); | 76 | int, int, intel_clock_t *); |
| 77 | }; | 77 | }; |
| 78 | 78 | ||
| 79 | #define I8XX_DOT_MIN 25000 | ||
| 80 | #define I8XX_DOT_MAX 350000 | ||
| 81 | #define I8XX_VCO_MIN 930000 | ||
| 82 | #define I8XX_VCO_MAX 1400000 | ||
| 83 | #define I8XX_N_MIN 3 | ||
| 84 | #define I8XX_N_MAX 16 | ||
| 85 | #define I8XX_M_MIN 96 | ||
| 86 | #define I8XX_M_MAX 140 | ||
| 87 | #define I8XX_M1_MIN 18 | ||
| 88 | #define I8XX_M1_MAX 26 | ||
| 89 | #define I8XX_M2_MIN 6 | ||
| 90 | #define I8XX_M2_MAX 16 | ||
| 91 | #define I8XX_P_MIN 4 | ||
| 92 | #define I8XX_P_MAX 128 | ||
| 93 | #define I8XX_P1_MIN 2 | ||
| 94 | #define I8XX_P1_MAX 33 | ||
| 95 | #define I8XX_P1_LVDS_MIN 1 | ||
| 96 | #define I8XX_P1_LVDS_MAX 6 | ||
| 97 | #define I8XX_P2_SLOW 4 | ||
| 98 | #define I8XX_P2_FAST 2 | ||
| 99 | #define I8XX_P2_LVDS_SLOW 14 | ||
| 100 | #define I8XX_P2_LVDS_FAST 7 | ||
| 101 | #define I8XX_P2_SLOW_LIMIT 165000 | ||
| 102 | |||
| 103 | #define I9XX_DOT_MIN 20000 | ||
| 104 | #define I9XX_DOT_MAX 400000 | ||
| 105 | #define I9XX_VCO_MIN 1400000 | ||
| 106 | #define I9XX_VCO_MAX 2800000 | ||
| 107 | #define PINEVIEW_VCO_MIN 1700000 | ||
| 108 | #define PINEVIEW_VCO_MAX 3500000 | ||
| 109 | #define I9XX_N_MIN 1 | ||
| 110 | #define I9XX_N_MAX 6 | ||
| 111 | /* Pineview's Ncounter is a ring counter */ | ||
| 112 | #define PINEVIEW_N_MIN 3 | ||
| 113 | #define PINEVIEW_N_MAX 6 | ||
| 114 | #define I9XX_M_MIN 70 | ||
| 115 | #define I9XX_M_MAX 120 | ||
| 116 | #define PINEVIEW_M_MIN 2 | ||
| 117 | #define PINEVIEW_M_MAX 256 | ||
| 118 | #define I9XX_M1_MIN 10 | ||
| 119 | #define I9XX_M1_MAX 22 | ||
| 120 | #define I9XX_M2_MIN 5 | ||
| 121 | #define I9XX_M2_MAX 9 | ||
| 122 | /* Pineview M1 is reserved, and must be 0 */ | ||
| 123 | #define PINEVIEW_M1_MIN 0 | ||
| 124 | #define PINEVIEW_M1_MAX 0 | ||
| 125 | #define PINEVIEW_M2_MIN 0 | ||
| 126 | #define PINEVIEW_M2_MAX 254 | ||
| 127 | #define I9XX_P_SDVO_DAC_MIN 5 | ||
| 128 | #define I9XX_P_SDVO_DAC_MAX 80 | ||
| 129 | #define I9XX_P_LVDS_MIN 7 | ||
| 130 | #define I9XX_P_LVDS_MAX 98 | ||
| 131 | #define PINEVIEW_P_LVDS_MIN 7 | ||
| 132 | #define PINEVIEW_P_LVDS_MAX 112 | ||
| 133 | #define I9XX_P1_MIN 1 | ||
| 134 | #define I9XX_P1_MAX 8 | ||
| 135 | #define I9XX_P2_SDVO_DAC_SLOW 10 | ||
| 136 | #define I9XX_P2_SDVO_DAC_FAST 5 | ||
| 137 | #define I9XX_P2_SDVO_DAC_SLOW_LIMIT 200000 | ||
| 138 | #define I9XX_P2_LVDS_SLOW 14 | ||
| 139 | #define I9XX_P2_LVDS_FAST 7 | ||
| 140 | #define I9XX_P2_LVDS_SLOW_LIMIT 112000 | ||
| 141 | |||
| 142 | /*The parameter is for SDVO on G4x platform*/ | ||
| 143 | #define G4X_DOT_SDVO_MIN 25000 | ||
| 144 | #define G4X_DOT_SDVO_MAX 270000 | ||
| 145 | #define G4X_VCO_MIN 1750000 | ||
| 146 | #define G4X_VCO_MAX 3500000 | ||
| 147 | #define G4X_N_SDVO_MIN 1 | ||
| 148 | #define G4X_N_SDVO_MAX 4 | ||
| 149 | #define G4X_M_SDVO_MIN 104 | ||
| 150 | #define G4X_M_SDVO_MAX 138 | ||
| 151 | #define G4X_M1_SDVO_MIN 17 | ||
| 152 | #define G4X_M1_SDVO_MAX 23 | ||
| 153 | #define G4X_M2_SDVO_MIN 5 | ||
| 154 | #define G4X_M2_SDVO_MAX 11 | ||
| 155 | #define G4X_P_SDVO_MIN 10 | ||
| 156 | #define G4X_P_SDVO_MAX 30 | ||
| 157 | #define G4X_P1_SDVO_MIN 1 | ||
| 158 | #define G4X_P1_SDVO_MAX 3 | ||
| 159 | #define G4X_P2_SDVO_SLOW 10 | ||
| 160 | #define G4X_P2_SDVO_FAST 10 | ||
| 161 | #define G4X_P2_SDVO_LIMIT 270000 | ||
| 162 | |||
| 163 | /*The parameter is for HDMI_DAC on G4x platform*/ | ||
| 164 | #define G4X_DOT_HDMI_DAC_MIN 22000 | ||
| 165 | #define G4X_DOT_HDMI_DAC_MAX 400000 | ||
| 166 | #define G4X_N_HDMI_DAC_MIN 1 | ||
| 167 | #define G4X_N_HDMI_DAC_MAX 4 | ||
| 168 | #define G4X_M_HDMI_DAC_MIN 104 | ||
| 169 | #define G4X_M_HDMI_DAC_MAX 138 | ||
| 170 | #define G4X_M1_HDMI_DAC_MIN 16 | ||
| 171 | #define G4X_M1_HDMI_DAC_MAX 23 | ||
| 172 | #define G4X_M2_HDMI_DAC_MIN 5 | ||
| 173 | #define G4X_M2_HDMI_DAC_MAX 11 | ||
| 174 | #define G4X_P_HDMI_DAC_MIN 5 | ||
| 175 | #define G4X_P_HDMI_DAC_MAX 80 | ||
| 176 | #define G4X_P1_HDMI_DAC_MIN 1 | ||
| 177 | #define G4X_P1_HDMI_DAC_MAX 8 | ||
| 178 | #define G4X_P2_HDMI_DAC_SLOW 10 | ||
| 179 | #define G4X_P2_HDMI_DAC_FAST 5 | ||
| 180 | #define G4X_P2_HDMI_DAC_LIMIT 165000 | ||
| 181 | |||
| 182 | /*The parameter is for SINGLE_CHANNEL_LVDS on G4x platform*/ | ||
| 183 | #define G4X_DOT_SINGLE_CHANNEL_LVDS_MIN 20000 | ||
| 184 | #define G4X_DOT_SINGLE_CHANNEL_LVDS_MAX 115000 | ||
| 185 | #define G4X_N_SINGLE_CHANNEL_LVDS_MIN 1 | ||
| 186 | #define G4X_N_SINGLE_CHANNEL_LVDS_MAX 3 | ||
| 187 | #define G4X_M_SINGLE_CHANNEL_LVDS_MIN 104 | ||
| 188 | #define G4X_M_SINGLE_CHANNEL_LVDS_MAX 138 | ||
| 189 | #define G4X_M1_SINGLE_CHANNEL_LVDS_MIN 17 | ||
| 190 | #define G4X_M1_SINGLE_CHANNEL_LVDS_MAX 23 | ||
| 191 | #define G4X_M2_SINGLE_CHANNEL_LVDS_MIN 5 | ||
| 192 | #define G4X_M2_SINGLE_CHANNEL_LVDS_MAX 11 | ||
| 193 | #define G4X_P_SINGLE_CHANNEL_LVDS_MIN 28 | ||
| 194 | #define G4X_P_SINGLE_CHANNEL_LVDS_MAX 112 | ||
| 195 | #define G4X_P1_SINGLE_CHANNEL_LVDS_MIN 2 | ||
| 196 | #define G4X_P1_SINGLE_CHANNEL_LVDS_MAX 8 | ||
| 197 | #define G4X_P2_SINGLE_CHANNEL_LVDS_SLOW 14 | ||
| 198 | #define G4X_P2_SINGLE_CHANNEL_LVDS_FAST 14 | ||
| 199 | #define G4X_P2_SINGLE_CHANNEL_LVDS_LIMIT 0 | ||
| 200 | |||
| 201 | /*The parameter is for DUAL_CHANNEL_LVDS on G4x platform*/ | ||
| 202 | #define G4X_DOT_DUAL_CHANNEL_LVDS_MIN 80000 | ||
| 203 | #define G4X_DOT_DUAL_CHANNEL_LVDS_MAX 224000 | ||
| 204 | #define G4X_N_DUAL_CHANNEL_LVDS_MIN 1 | ||
| 205 | #define G4X_N_DUAL_CHANNEL_LVDS_MAX 3 | ||
| 206 | #define G4X_M_DUAL_CHANNEL_LVDS_MIN 104 | ||
| 207 | #define G4X_M_DUAL_CHANNEL_LVDS_MAX 138 | ||
| 208 | #define G4X_M1_DUAL_CHANNEL_LVDS_MIN 17 | ||
| 209 | #define G4X_M1_DUAL_CHANNEL_LVDS_MAX 23 | ||
| 210 | #define G4X_M2_DUAL_CHANNEL_LVDS_MIN 5 | ||
| 211 | #define G4X_M2_DUAL_CHANNEL_LVDS_MAX 11 | ||
| 212 | #define G4X_P_DUAL_CHANNEL_LVDS_MIN 14 | ||
| 213 | #define G4X_P_DUAL_CHANNEL_LVDS_MAX 42 | ||
| 214 | #define G4X_P1_DUAL_CHANNEL_LVDS_MIN 2 | ||
| 215 | #define G4X_P1_DUAL_CHANNEL_LVDS_MAX 6 | ||
| 216 | #define G4X_P2_DUAL_CHANNEL_LVDS_SLOW 7 | ||
| 217 | #define G4X_P2_DUAL_CHANNEL_LVDS_FAST 7 | ||
| 218 | #define G4X_P2_DUAL_CHANNEL_LVDS_LIMIT 0 | ||
| 219 | |||
| 220 | /*The parameter is for DISPLAY PORT on G4x platform*/ | ||
| 221 | #define G4X_DOT_DISPLAY_PORT_MIN 161670 | ||
| 222 | #define G4X_DOT_DISPLAY_PORT_MAX 227000 | ||
| 223 | #define G4X_N_DISPLAY_PORT_MIN 1 | ||
| 224 | #define G4X_N_DISPLAY_PORT_MAX 2 | ||
| 225 | #define G4X_M_DISPLAY_PORT_MIN 97 | ||
| 226 | #define G4X_M_DISPLAY_PORT_MAX 108 | ||
| 227 | #define G4X_M1_DISPLAY_PORT_MIN 0x10 | ||
| 228 | #define G4X_M1_DISPLAY_PORT_MAX 0x12 | ||
| 229 | #define G4X_M2_DISPLAY_PORT_MIN 0x05 | ||
| 230 | #define G4X_M2_DISPLAY_PORT_MAX 0x06 | ||
| 231 | #define G4X_P_DISPLAY_PORT_MIN 10 | ||
| 232 | #define G4X_P_DISPLAY_PORT_MAX 20 | ||
| 233 | #define G4X_P1_DISPLAY_PORT_MIN 1 | ||
| 234 | #define G4X_P1_DISPLAY_PORT_MAX 2 | ||
| 235 | #define G4X_P2_DISPLAY_PORT_SLOW 10 | ||
| 236 | #define G4X_P2_DISPLAY_PORT_FAST 10 | ||
| 237 | #define G4X_P2_DISPLAY_PORT_LIMIT 0 | ||
| 238 | |||
| 239 | /* Ironlake / Sandybridge */ | ||
| 240 | /* as we calculate clock using (register_value + 2) for | ||
| 241 | N/M1/M2, so here the range value for them is (actual_value-2). | ||
| 242 | */ | ||
| 243 | #define IRONLAKE_DOT_MIN 25000 | ||
| 244 | #define IRONLAKE_DOT_MAX 350000 | ||
| 245 | #define IRONLAKE_VCO_MIN 1760000 | ||
| 246 | #define IRONLAKE_VCO_MAX 3510000 | ||
| 247 | #define IRONLAKE_M1_MIN 12 | ||
| 248 | #define IRONLAKE_M1_MAX 22 | ||
| 249 | #define IRONLAKE_M2_MIN 5 | ||
| 250 | #define IRONLAKE_M2_MAX 9 | ||
| 251 | #define IRONLAKE_P2_DOT_LIMIT 225000 /* 225Mhz */ | ||
| 252 | |||
| 253 | /* We have parameter ranges for different type of outputs. */ | ||
| 254 | |||
| 255 | /* DAC & HDMI Refclk 120Mhz */ | ||
| 256 | #define IRONLAKE_DAC_N_MIN 1 | ||
| 257 | #define IRONLAKE_DAC_N_MAX 5 | ||
| 258 | #define IRONLAKE_DAC_M_MIN 79 | ||
| 259 | #define IRONLAKE_DAC_M_MAX 127 | ||
| 260 | #define IRONLAKE_DAC_P_MIN 5 | ||
| 261 | #define IRONLAKE_DAC_P_MAX 80 | ||
| 262 | #define IRONLAKE_DAC_P1_MIN 1 | ||
| 263 | #define IRONLAKE_DAC_P1_MAX 8 | ||
| 264 | #define IRONLAKE_DAC_P2_SLOW 10 | ||
| 265 | #define IRONLAKE_DAC_P2_FAST 5 | ||
| 266 | |||
| 267 | /* LVDS single-channel 120Mhz refclk */ | ||
| 268 | #define IRONLAKE_LVDS_S_N_MIN 1 | ||
| 269 | #define IRONLAKE_LVDS_S_N_MAX 3 | ||
| 270 | #define IRONLAKE_LVDS_S_M_MIN 79 | ||
| 271 | #define IRONLAKE_LVDS_S_M_MAX 118 | ||
| 272 | #define IRONLAKE_LVDS_S_P_MIN 28 | ||
| 273 | #define IRONLAKE_LVDS_S_P_MAX 112 | ||
| 274 | #define IRONLAKE_LVDS_S_P1_MIN 2 | ||
| 275 | #define IRONLAKE_LVDS_S_P1_MAX 8 | ||
| 276 | #define IRONLAKE_LVDS_S_P2_SLOW 14 | ||
| 277 | #define IRONLAKE_LVDS_S_P2_FAST 14 | ||
| 278 | |||
| 279 | /* LVDS dual-channel 120Mhz refclk */ | ||
| 280 | #define IRONLAKE_LVDS_D_N_MIN 1 | ||
| 281 | #define IRONLAKE_LVDS_D_N_MAX 3 | ||
| 282 | #define IRONLAKE_LVDS_D_M_MIN 79 | ||
| 283 | #define IRONLAKE_LVDS_D_M_MAX 127 | ||
| 284 | #define IRONLAKE_LVDS_D_P_MIN 14 | ||
| 285 | #define IRONLAKE_LVDS_D_P_MAX 56 | ||
| 286 | #define IRONLAKE_LVDS_D_P1_MIN 2 | ||
| 287 | #define IRONLAKE_LVDS_D_P1_MAX 8 | ||
| 288 | #define IRONLAKE_LVDS_D_P2_SLOW 7 | ||
| 289 | #define IRONLAKE_LVDS_D_P2_FAST 7 | ||
| 290 | |||
| 291 | /* LVDS single-channel 100Mhz refclk */ | ||
| 292 | #define IRONLAKE_LVDS_S_SSC_N_MIN 1 | ||
| 293 | #define IRONLAKE_LVDS_S_SSC_N_MAX 2 | ||
| 294 | #define IRONLAKE_LVDS_S_SSC_M_MIN 79 | ||
| 295 | #define IRONLAKE_LVDS_S_SSC_M_MAX 126 | ||
| 296 | #define IRONLAKE_LVDS_S_SSC_P_MIN 28 | ||
| 297 | #define IRONLAKE_LVDS_S_SSC_P_MAX 112 | ||
| 298 | #define IRONLAKE_LVDS_S_SSC_P1_MIN 2 | ||
| 299 | #define IRONLAKE_LVDS_S_SSC_P1_MAX 8 | ||
| 300 | #define IRONLAKE_LVDS_S_SSC_P2_SLOW 14 | ||
| 301 | #define IRONLAKE_LVDS_S_SSC_P2_FAST 14 | ||
| 302 | |||
| 303 | /* LVDS dual-channel 100Mhz refclk */ | ||
| 304 | #define IRONLAKE_LVDS_D_SSC_N_MIN 1 | ||
| 305 | #define IRONLAKE_LVDS_D_SSC_N_MAX 3 | ||
| 306 | #define IRONLAKE_LVDS_D_SSC_M_MIN 79 | ||
| 307 | #define IRONLAKE_LVDS_D_SSC_M_MAX 126 | ||
| 308 | #define IRONLAKE_LVDS_D_SSC_P_MIN 14 | ||
| 309 | #define IRONLAKE_LVDS_D_SSC_P_MAX 42 | ||
| 310 | #define IRONLAKE_LVDS_D_SSC_P1_MIN 2 | ||
| 311 | #define IRONLAKE_LVDS_D_SSC_P1_MAX 6 | ||
| 312 | #define IRONLAKE_LVDS_D_SSC_P2_SLOW 7 | ||
| 313 | #define IRONLAKE_LVDS_D_SSC_P2_FAST 7 | ||
| 314 | |||
| 315 | /* DisplayPort */ | ||
| 316 | #define IRONLAKE_DP_N_MIN 1 | ||
| 317 | #define IRONLAKE_DP_N_MAX 2 | ||
| 318 | #define IRONLAKE_DP_M_MIN 81 | ||
| 319 | #define IRONLAKE_DP_M_MAX 90 | ||
| 320 | #define IRONLAKE_DP_P_MIN 10 | ||
| 321 | #define IRONLAKE_DP_P_MAX 20 | ||
| 322 | #define IRONLAKE_DP_P2_FAST 10 | ||
| 323 | #define IRONLAKE_DP_P2_SLOW 10 | ||
| 324 | #define IRONLAKE_DP_P2_LIMIT 0 | ||
| 325 | #define IRONLAKE_DP_P1_MIN 1 | ||
| 326 | #define IRONLAKE_DP_P1_MAX 2 | ||
| 327 | |||
| 328 | /* FDI */ | 79 | /* FDI */ |
| 329 | #define IRONLAKE_FDI_FREQ 2700000 /* in kHz for mode->clock */ | 80 | #define IRONLAKE_FDI_FREQ 2700000 /* in kHz for mode->clock */ |
| 330 | 81 | ||
| @@ -353,292 +104,253 @@ intel_fdi_link_freq(struct drm_device *dev) | |||
| 353 | } | 104 | } |
| 354 | 105 | ||
| 355 | static const intel_limit_t intel_limits_i8xx_dvo = { | 106 | static const intel_limit_t intel_limits_i8xx_dvo = { |
| 356 | .dot = { .min = I8XX_DOT_MIN, .max = I8XX_DOT_MAX }, | 107 | .dot = { .min = 25000, .max = 350000 }, |
| 357 | .vco = { .min = I8XX_VCO_MIN, .max = I8XX_VCO_MAX }, | 108 | .vco = { .min = 930000, .max = 1400000 }, |
| 358 | .n = { .min = I8XX_N_MIN, .max = I8XX_N_MAX }, | 109 | .n = { .min = 3, .max = 16 }, |
| 359 | .m = { .min = I8XX_M_MIN, .max = I8XX_M_MAX }, | 110 | .m = { .min = 96, .max = 140 }, |
| 360 | .m1 = { .min = I8XX_M1_MIN, .max = I8XX_M1_MAX }, | 111 | .m1 = { .min = 18, .max = 26 }, |
| 361 | .m2 = { .min = I8XX_M2_MIN, .max = I8XX_M2_MAX }, | 112 | .m2 = { .min = 6, .max = 16 }, |
| 362 | .p = { .min = I8XX_P_MIN, .max = I8XX_P_MAX }, | 113 | .p = { .min = 4, .max = 128 }, |
| 363 | .p1 = { .min = I8XX_P1_MIN, .max = I8XX_P1_MAX }, | 114 | .p1 = { .min = 2, .max = 33 }, |
| 364 | .p2 = { .dot_limit = I8XX_P2_SLOW_LIMIT, | 115 | .p2 = { .dot_limit = 165000, |
| 365 | .p2_slow = I8XX_P2_SLOW, .p2_fast = I8XX_P2_FAST }, | 116 | .p2_slow = 4, .p2_fast = 2 }, |
| 366 | .find_pll = intel_find_best_PLL, | 117 | .find_pll = intel_find_best_PLL, |
| 367 | }; | 118 | }; |
| 368 | 119 | ||
| 369 | static const intel_limit_t intel_limits_i8xx_lvds = { | 120 | static const intel_limit_t intel_limits_i8xx_lvds = { |
| 370 | .dot = { .min = I8XX_DOT_MIN, .max = I8XX_DOT_MAX }, | 121 | .dot = { .min = 25000, .max = 350000 }, |
| 371 | .vco = { .min = I8XX_VCO_MIN, .max = I8XX_VCO_MAX }, | 122 | .vco = { .min = 930000, .max = 1400000 }, |
| 372 | .n = { .min = I8XX_N_MIN, .max = I8XX_N_MAX }, | 123 | .n = { .min = 3, .max = 16 }, |
| 373 | .m = { .min = I8XX_M_MIN, .max = I8XX_M_MAX }, | 124 | .m = { .min = 96, .max = 140 }, |
| 374 | .m1 = { .min = I8XX_M1_MIN, .max = I8XX_M1_MAX }, | 125 | .m1 = { .min = 18, .max = 26 }, |
| 375 | .m2 = { .min = I8XX_M2_MIN, .max = I8XX_M2_MAX }, | 126 | .m2 = { .min = 6, .max = 16 }, |
| 376 | .p = { .min = I8XX_P_MIN, .max = I8XX_P_MAX }, | 127 | .p = { .min = 4, .max = 128 }, |
| 377 | .p1 = { .min = I8XX_P1_LVDS_MIN, .max = I8XX_P1_LVDS_MAX }, | 128 | .p1 = { .min = 1, .max = 6 }, |
| 378 | .p2 = { .dot_limit = I8XX_P2_SLOW_LIMIT, | 129 | .p2 = { .dot_limit = 165000, |
| 379 | .p2_slow = I8XX_P2_LVDS_SLOW, .p2_fast = I8XX_P2_LVDS_FAST }, | 130 | .p2_slow = 14, .p2_fast = 7 }, |
| 380 | .find_pll = intel_find_best_PLL, | 131 | .find_pll = intel_find_best_PLL, |
| 381 | }; | 132 | }; |
| 382 | 133 | ||
| 383 | static const intel_limit_t intel_limits_i9xx_sdvo = { | 134 | static const intel_limit_t intel_limits_i9xx_sdvo = { |
| 384 | .dot = { .min = I9XX_DOT_MIN, .max = I9XX_DOT_MAX }, | 135 | .dot = { .min = 20000, .max = 400000 }, |
| 385 | .vco = { .min = I9XX_VCO_MIN, .max = I9XX_VCO_MAX }, | 136 | .vco = { .min = 1400000, .max = 2800000 }, |
| 386 | .n = { .min = I9XX_N_MIN, .max = I9XX_N_MAX }, | 137 | .n = { .min = 1, .max = 6 }, |
| 387 | .m = { .min = I9XX_M_MIN, .max = I9XX_M_MAX }, | 138 | .m = { .min = 70, .max = 120 }, |
| 388 | .m1 = { .min = I9XX_M1_MIN, .max = I9XX_M1_MAX }, | 139 | .m1 = { .min = 10, .max = 22 }, |
| 389 | .m2 = { .min = I9XX_M2_MIN, .max = I9XX_M2_MAX }, | 140 | .m2 = { .min = 5, .max = 9 }, |
| 390 | .p = { .min = I9XX_P_SDVO_DAC_MIN, .max = I9XX_P_SDVO_DAC_MAX }, | 141 | .p = { .min = 5, .max = 80 }, |
| 391 | .p1 = { .min = I9XX_P1_MIN, .max = I9XX_P1_MAX }, | 142 | .p1 = { .min = 1, .max = 8 }, |
| 392 | .p2 = { .dot_limit = I9XX_P2_SDVO_DAC_SLOW_LIMIT, | 143 | .p2 = { .dot_limit = 200000, |
| 393 | .p2_slow = I9XX_P2_SDVO_DAC_SLOW, .p2_fast = I9XX_P2_SDVO_DAC_FAST }, | 144 | .p2_slow = 10, .p2_fast = 5 }, |
| 394 | .find_pll = intel_find_best_PLL, | 145 | .find_pll = intel_find_best_PLL, |
| 395 | }; | 146 | }; |
| 396 | 147 | ||
| 397 | static const intel_limit_t intel_limits_i9xx_lvds = { | 148 | static const intel_limit_t intel_limits_i9xx_lvds = { |
| 398 | .dot = { .min = I9XX_DOT_MIN, .max = I9XX_DOT_MAX }, | 149 | .dot = { .min = 20000, .max = 400000 }, |
| 399 | .vco = { .min = I9XX_VCO_MIN, .max = I9XX_VCO_MAX }, | 150 | .vco = { .min = 1400000, .max = 2800000 }, |
| 400 | .n = { .min = I9XX_N_MIN, .max = I9XX_N_MAX }, | 151 | .n = { .min = 1, .max = 6 }, |
| 401 | .m = { .min = I9XX_M_MIN, .max = I9XX_M_MAX }, | 152 | .m = { .min = 70, .max = 120 }, |
| 402 | .m1 = { .min = I9XX_M1_MIN, .max = I9XX_M1_MAX }, | 153 | .m1 = { .min = 10, .max = 22 }, |
| 403 | .m2 = { .min = I9XX_M2_MIN, .max = I9XX_M2_MAX }, | 154 | .m2 = { .min = 5, .max = 9 }, |
| 404 | .p = { .min = I9XX_P_LVDS_MIN, .max = I9XX_P_LVDS_MAX }, | 155 | .p = { .min = 7, .max = 98 }, |
| 405 | .p1 = { .min = I9XX_P1_MIN, .max = I9XX_P1_MAX }, | 156 | .p1 = { .min = 1, .max = 8 }, |
| 406 | /* The single-channel range is 25-112Mhz, and dual-channel | 157 | .p2 = { .dot_limit = 112000, |
| 407 | * is 80-224Mhz. Prefer single channel as much as possible. | 158 | .p2_slow = 14, .p2_fast = 7 }, |
| 408 | */ | ||
| 409 | .p2 = { .dot_limit = I9XX_P2_LVDS_SLOW_LIMIT, | ||
| 410 | .p2_slow = I9XX_P2_LVDS_SLOW, .p2_fast = I9XX_P2_LVDS_FAST }, | ||
| 411 | .find_pll = intel_find_best_PLL, | 159 | .find_pll = intel_find_best_PLL, |
| 412 | }; | 160 | }; |
| 413 | 161 | ||
| 414 | /* below parameter and function is for G4X Chipset Family*/ | 162 | |
| 415 | static const intel_limit_t intel_limits_g4x_sdvo = { | 163 | static const intel_limit_t intel_limits_g4x_sdvo = { |
| 416 | .dot = { .min = G4X_DOT_SDVO_MIN, .max = G4X_DOT_SDVO_MAX }, | 164 | .dot = { .min = 25000, .max = 270000 }, |
| 417 | .vco = { .min = G4X_VCO_MIN, .max = G4X_VCO_MAX}, | 165 | .vco = { .min = 1750000, .max = 3500000}, |
| 418 | .n = { .min = G4X_N_SDVO_MIN, .max = G4X_N_SDVO_MAX }, | 166 | .n = { .min = 1, .max = 4 }, |
| 419 | .m = { .min = G4X_M_SDVO_MIN, .max = G4X_M_SDVO_MAX }, | 167 | .m = { .min = 104, .max = 138 }, |
| 420 | .m1 = { .min = G4X_M1_SDVO_MIN, .max = G4X_M1_SDVO_MAX }, | 168 | .m1 = { .min = 17, .max = 23 }, |
| 421 | .m2 = { .min = G4X_M2_SDVO_MIN, .max = G4X_M2_SDVO_MAX }, | 169 | .m2 = { .min = 5, .max = 11 }, |
| 422 | .p = { .min = G4X_P_SDVO_MIN, .max = G4X_P_SDVO_MAX }, | 170 | .p = { .min = 10, .max = 30 }, |
| 423 | .p1 = { .min = G4X_P1_SDVO_MIN, .max = G4X_P1_SDVO_MAX}, | 171 | .p1 = { .min = 1, .max = 3}, |
| 424 | .p2 = { .dot_limit = G4X_P2_SDVO_LIMIT, | 172 | .p2 = { .dot_limit = 270000, |
| 425 | .p2_slow = G4X_P2_SDVO_SLOW, | 173 | .p2_slow = 10, |
| 426 | .p2_fast = G4X_P2_SDVO_FAST | 174 | .p2_fast = 10 |
| 427 | }, | 175 | }, |
| 428 | .find_pll = intel_g4x_find_best_PLL, | 176 | .find_pll = intel_g4x_find_best_PLL, |
| 429 | }; | 177 | }; |
| 430 | 178 | ||
| 431 | static const intel_limit_t intel_limits_g4x_hdmi = { | 179 | static const intel_limit_t intel_limits_g4x_hdmi = { |
| 432 | .dot = { .min = G4X_DOT_HDMI_DAC_MIN, .max = G4X_DOT_HDMI_DAC_MAX }, | 180 | .dot = { .min = 22000, .max = 400000 }, |
| 433 | .vco = { .min = G4X_VCO_MIN, .max = G4X_VCO_MAX}, | 181 | .vco = { .min = 1750000, .max = 3500000}, |
| 434 | .n = { .min = G4X_N_HDMI_DAC_MIN, .max = G4X_N_HDMI_DAC_MAX }, | 182 | .n = { .min = 1, .max = 4 }, |
| 435 | .m = { .min = G4X_M_HDMI_DAC_MIN, .max = G4X_M_HDMI_DAC_MAX }, | 183 | .m = { .min = 104, .max = 138 }, |
| 436 | .m1 = { .min = G4X_M1_HDMI_DAC_MIN, .max = G4X_M1_HDMI_DAC_MAX }, | 184 | .m1 = { .min = 16, .max = 23 }, |
| 437 | .m2 = { .min = G4X_M2_HDMI_DAC_MIN, .max = G4X_M2_HDMI_DAC_MAX }, | 185 | .m2 = { .min = 5, .max = 11 }, |
| 438 | .p = { .min = G4X_P_HDMI_DAC_MIN, .max = G4X_P_HDMI_DAC_MAX }, | 186 | .p = { .min = 5, .max = 80 }, |
| 439 | .p1 = { .min = G4X_P1_HDMI_DAC_MIN, .max = G4X_P1_HDMI_DAC_MAX}, | 187 | .p1 = { .min = 1, .max = 8}, |
| 440 | .p2 = { .dot_limit = G4X_P2_HDMI_DAC_LIMIT, | 188 | .p2 = { .dot_limit = 165000, |
| 441 | .p2_slow = G4X_P2_HDMI_DAC_SLOW, | 189 | .p2_slow = 10, .p2_fast = 5 }, |
| 442 | .p2_fast = G4X_P2_HDMI_DAC_FAST | ||
| 443 | }, | ||
| 444 | .find_pll = intel_g4x_find_best_PLL, | 190 | .find_pll = intel_g4x_find_best_PLL, |
| 445 | }; | 191 | }; |
| 446 | 192 | ||
| 447 | static const intel_limit_t intel_limits_g4x_single_channel_lvds = { | 193 | static const intel_limit_t intel_limits_g4x_single_channel_lvds = { |
| 448 | .dot = { .min = G4X_DOT_SINGLE_CHANNEL_LVDS_MIN, | 194 | .dot = { .min = 20000, .max = 115000 }, |
| 449 | .max = G4X_DOT_SINGLE_CHANNEL_LVDS_MAX }, | 195 | .vco = { .min = 1750000, .max = 3500000 }, |
| 450 | .vco = { .min = G4X_VCO_MIN, | 196 | .n = { .min = 1, .max = 3 }, |
| 451 | .max = G4X_VCO_MAX }, | 197 | .m = { .min = 104, .max = 138 }, |
| 452 | .n = { .min = G4X_N_SINGLE_CHANNEL_LVDS_MIN, | 198 | .m1 = { .min = 17, .max = 23 }, |
| 453 | .max = G4X_N_SINGLE_CHANNEL_LVDS_MAX }, | 199 | .m2 = { .min = 5, .max = 11 }, |
| 454 | .m = { .min = G4X_M_SINGLE_CHANNEL_LVDS_MIN, | 200 | .p = { .min = 28, .max = 112 }, |
| 455 | .max = G4X_M_SINGLE_CHANNEL_LVDS_MAX }, | 201 | .p1 = { .min = 2, .max = 8 }, |
| 456 | .m1 = { .min = G4X_M1_SINGLE_CHANNEL_LVDS_MIN, | 202 | .p2 = { .dot_limit = 0, |
| 457 | .max = G4X_M1_SINGLE_CHANNEL_LVDS_MAX }, | 203 | .p2_slow = 14, .p2_fast = 14 |
| 458 | .m2 = { .min = G4X_M2_SINGLE_CHANNEL_LVDS_MIN, | ||
| 459 | .max = G4X_M2_SINGLE_CHANNEL_LVDS_MAX }, | ||
| 460 | .p = { .min = G4X_P_SINGLE_CHANNEL_LVDS_MIN, | ||
| 461 | .max = G4X_P_SINGLE_CHANNEL_LVDS_MAX }, | ||
| 462 | .p1 = { .min = G4X_P1_SINGLE_CHANNEL_LVDS_MIN, | ||
| 463 | .max = G4X_P1_SINGLE_CHANNEL_LVDS_MAX }, | ||
| 464 | .p2 = { .dot_limit = G4X_P2_SINGLE_CHANNEL_LVDS_LIMIT, | ||
| 465 | .p2_slow = G4X_P2_SINGLE_CHANNEL_LVDS_SLOW, | ||
| 466 | .p2_fast = G4X_P2_SINGLE_CHANNEL_LVDS_FAST | ||
| 467 | }, | 204 | }, |
| 468 | .find_pll = intel_g4x_find_best_PLL, | 205 | .find_pll = intel_g4x_find_best_PLL, |
| 469 | }; | 206 | }; |
| 470 | 207 | ||
| 471 | static const intel_limit_t intel_limits_g4x_dual_channel_lvds = { | 208 | static const intel_limit_t intel_limits_g4x_dual_channel_lvds = { |
| 472 | .dot = { .min = G4X_DOT_DUAL_CHANNEL_LVDS_MIN, | 209 | .dot = { .min = 80000, .max = 224000 }, |
| 473 | .max = G4X_DOT_DUAL_CHANNEL_LVDS_MAX }, | 210 | .vco = { .min = 1750000, .max = 3500000 }, |
| 474 | .vco = { .min = G4X_VCO_MIN, | 211 | .n = { .min = 1, .max = 3 }, |
| 475 | .max = G4X_VCO_MAX }, | 212 | .m = { .min = 104, .max = 138 }, |
| 476 | .n = { .min = G4X_N_DUAL_CHANNEL_LVDS_MIN, | 213 | .m1 = { .min = 17, .max = 23 }, |
| 477 | .max = G4X_N_DUAL_CHANNEL_LVDS_MAX }, | 214 | .m2 = { .min = 5, .max = 11 }, |
| 478 | .m = { .min = G4X_M_DUAL_CHANNEL_LVDS_MIN, | 215 | .p = { .min = 14, .max = 42 }, |
| 479 | .max = G4X_M_DUAL_CHANNEL_LVDS_MAX }, | 216 | .p1 = { .min = 2, .max = 6 }, |
| 480 | .m1 = { .min = G4X_M1_DUAL_CHANNEL_LVDS_MIN, | 217 | .p2 = { .dot_limit = 0, |
| 481 | .max = G4X_M1_DUAL_CHANNEL_LVDS_MAX }, | 218 | .p2_slow = 7, .p2_fast = 7 |
| 482 | .m2 = { .min = G4X_M2_DUAL_CHANNEL_LVDS_MIN, | ||
| 483 | .max = G4X_M2_DUAL_CHANNEL_LVDS_MAX }, | ||
| 484 | .p = { .min = G4X_P_DUAL_CHANNEL_LVDS_MIN, | ||
| 485 | .max = G4X_P_DUAL_CHANNEL_LVDS_MAX }, | ||
| 486 | .p1 = { .min = G4X_P1_DUAL_CHANNEL_LVDS_MIN, | ||
| 487 | .max = G4X_P1_DUAL_CHANNEL_LVDS_MAX }, | ||
| 488 | .p2 = { .dot_limit = G4X_P2_DUAL_CHANNEL_LVDS_LIMIT, | ||
| 489 | .p2_slow = G4X_P2_DUAL_CHANNEL_LVDS_SLOW, | ||
| 490 | .p2_fast = G4X_P2_DUAL_CHANNEL_LVDS_FAST | ||
| 491 | }, | 219 | }, |
| 492 | .find_pll = intel_g4x_find_best_PLL, | 220 | .find_pll = intel_g4x_find_best_PLL, |
| 493 | }; | 221 | }; |
| 494 | 222 | ||
| 495 | static const intel_limit_t intel_limits_g4x_display_port = { | 223 | static const intel_limit_t intel_limits_g4x_display_port = { |
| 496 | .dot = { .min = G4X_DOT_DISPLAY_PORT_MIN, | 224 | .dot = { .min = 161670, .max = 227000 }, |
| 497 | .max = G4X_DOT_DISPLAY_PORT_MAX }, | 225 | .vco = { .min = 1750000, .max = 3500000}, |
| 498 | .vco = { .min = G4X_VCO_MIN, | 226 | .n = { .min = 1, .max = 2 }, |
| 499 | .max = G4X_VCO_MAX}, | 227 | .m = { .min = 97, .max = 108 }, |
| 500 | .n = { .min = G4X_N_DISPLAY_PORT_MIN, | 228 | .m1 = { .min = 0x10, .max = 0x12 }, |
| 501 | .max = G4X_N_DISPLAY_PORT_MAX }, | 229 | .m2 = { .min = 0x05, .max = 0x06 }, |
| 502 | .m = { .min = G4X_M_DISPLAY_PORT_MIN, | 230 | .p = { .min = 10, .max = 20 }, |
| 503 | .max = G4X_M_DISPLAY_PORT_MAX }, | 231 | .p1 = { .min = 1, .max = 2}, |
| 504 | .m1 = { .min = G4X_M1_DISPLAY_PORT_MIN, | 232 | .p2 = { .dot_limit = 0, |
| 505 | .max = G4X_M1_DISPLAY_PORT_MAX }, | 233 | .p2_slow = 10, .p2_fast = 10 }, |
| 506 | .m2 = { .min = G4X_M2_DISPLAY_PORT_MIN, | ||
| 507 | .max = G4X_M2_DISPLAY_PORT_MAX }, | ||
| 508 | .p = { .min = G4X_P_DISPLAY_PORT_MIN, | ||
| 509 | .max = G4X_P_DISPLAY_PORT_MAX }, | ||
| 510 | .p1 = { .min = G4X_P1_DISPLAY_PORT_MIN, | ||
| 511 | .max = G4X_P1_DISPLAY_PORT_MAX}, | ||
| 512 | .p2 = { .dot_limit = G4X_P2_DISPLAY_PORT_LIMIT, | ||
| 513 | .p2_slow = G4X_P2_DISPLAY_PORT_SLOW, | ||
| 514 | .p2_fast = G4X_P2_DISPLAY_PORT_FAST }, | ||
| 515 | .find_pll = intel_find_pll_g4x_dp, | 234 | .find_pll = intel_find_pll_g4x_dp, |
| 516 | }; | 235 | }; |
| 517 | 236 | ||
| 518 | static const intel_limit_t intel_limits_pineview_sdvo = { | 237 | static const intel_limit_t intel_limits_pineview_sdvo = { |
| 519 | .dot = { .min = I9XX_DOT_MIN, .max = I9XX_DOT_MAX}, | 238 | .dot = { .min = 20000, .max = 400000}, |
| 520 | .vco = { .min = PINEVIEW_VCO_MIN, .max = PINEVIEW_VCO_MAX }, | 239 | .vco = { .min = 1700000, .max = 3500000 }, |
| 521 | .n = { .min = PINEVIEW_N_MIN, .max = PINEVIEW_N_MAX }, | 240 | /* Pineview's Ncounter is a ring counter */ |
| 522 | .m = { .min = PINEVIEW_M_MIN, .max = PINEVIEW_M_MAX }, | 241 | .n = { .min = 3, .max = 6 }, |
| 523 | .m1 = { .min = PINEVIEW_M1_MIN, .max = PINEVIEW_M1_MAX }, | 242 | .m = { .min = 2, .max = 256 }, |
| 524 | .m2 = { .min = PINEVIEW_M2_MIN, .max = PINEVIEW_M2_MAX }, | 243 | /* Pineview only has one combined m divider, which we treat as m2. */ |
| 525 | .p = { .min = I9XX_P_SDVO_DAC_MIN, .max = I9XX_P_SDVO_DAC_MAX }, | 244 | .m1 = { .min = 0, .max = 0 }, |
| 526 | .p1 = { .min = I9XX_P1_MIN, .max = I9XX_P1_MAX }, | 245 | .m2 = { .min = 0, .max = 254 }, |
| 527 | .p2 = { .dot_limit = I9XX_P2_SDVO_DAC_SLOW_LIMIT, | 246 | .p = { .min = 5, .max = 80 }, |
| 528 | .p2_slow = I9XX_P2_SDVO_DAC_SLOW, .p2_fast = I9XX_P2_SDVO_DAC_FAST }, | 247 | .p1 = { .min = 1, .max = 8 }, |
| 248 | .p2 = { .dot_limit = 200000, | ||
| 249 | .p2_slow = 10, .p2_fast = 5 }, | ||
| 529 | .find_pll = intel_find_best_PLL, | 250 | .find_pll = intel_find_best_PLL, |
| 530 | }; | 251 | }; |
| 531 | 252 | ||
| 532 | static const intel_limit_t intel_limits_pineview_lvds = { | 253 | static const intel_limit_t intel_limits_pineview_lvds = { |
| 533 | .dot = { .min = I9XX_DOT_MIN, .max = I9XX_DOT_MAX }, | 254 | .dot = { .min = 20000, .max = 400000 }, |
| 534 | .vco = { .min = PINEVIEW_VCO_MIN, .max = PINEVIEW_VCO_MAX }, | 255 | .vco = { .min = 1700000, .max = 3500000 }, |
| 535 | .n = { .min = PINEVIEW_N_MIN, .max = PINEVIEW_N_MAX }, | 256 | .n = { .min = 3, .max = 6 }, |
| 536 | .m = { .min = PINEVIEW_M_MIN, .max = PINEVIEW_M_MAX }, | 257 | .m = { .min = 2, .max = 256 }, |
| 537 | .m1 = { .min = PINEVIEW_M1_MIN, .max = PINEVIEW_M1_MAX }, | 258 | .m1 = { .min = 0, .max = 0 }, |
| 538 | .m2 = { .min = PINEVIEW_M2_MIN, .max = PINEVIEW_M2_MAX }, | 259 | .m2 = { .min = 0, .max = 254 }, |
| 539 | .p = { .min = PINEVIEW_P_LVDS_MIN, .max = PINEVIEW_P_LVDS_MAX }, | 260 | .p = { .min = 7, .max = 112 }, |
| 540 | .p1 = { .min = I9XX_P1_MIN, .max = I9XX_P1_MAX }, | 261 | .p1 = { .min = 1, .max = 8 }, |
| 541 | /* Pineview only supports single-channel mode. */ | 262 | .p2 = { .dot_limit = 112000, |
| 542 | .p2 = { .dot_limit = I9XX_P2_LVDS_SLOW_LIMIT, | 263 | .p2_slow = 14, .p2_fast = 14 }, |
| 543 | .p2_slow = I9XX_P2_LVDS_SLOW, .p2_fast = I9XX_P2_LVDS_SLOW }, | ||
| 544 | .find_pll = intel_find_best_PLL, | 264 | .find_pll = intel_find_best_PLL, |
| 545 | }; | 265 | }; |
| 546 | 266 | ||
| 267 | /* Ironlake / Sandybridge | ||
| 268 | * | ||
| 269 | * We calculate clock using (register_value + 2) for N/M1/M2, so here | ||
| 270 | * the range value for them is (actual_value - 2). | ||
| 271 | */ | ||
| 547 | static const intel_limit_t intel_limits_ironlake_dac = { | 272 | static const intel_limit_t intel_limits_ironlake_dac = { |
| 548 | .dot = { .min = IRONLAKE_DOT_MIN, .max = IRONLAKE_DOT_MAX }, | 273 | .dot = { .min = 25000, .max = 350000 }, |
| 549 | .vco = { .min = IRONLAKE_VCO_MIN, .max = IRONLAKE_VCO_MAX }, | 274 | .vco = { .min = 1760000, .max = 3510000 }, |
| 550 | .n = { .min = IRONLAKE_DAC_N_MIN, .max = IRONLAKE_DAC_N_MAX }, | 275 | .n = { .min = 1, .max = 5 }, |
| 551 | .m = { .min = IRONLAKE_DAC_M_MIN, .max = IRONLAKE_DAC_M_MAX }, | 276 | .m = { .min = 79, .max = 127 }, |
| 552 | .m1 = { .min = IRONLAKE_M1_MIN, .max = IRONLAKE_M1_MAX }, | 277 | .m1 = { .min = 12, .max = 22 }, |
| 553 | .m2 = { .min = IRONLAKE_M2_MIN, .max = IRONLAKE_M2_MAX }, | 278 | .m2 = { .min = 5, .max = 9 }, |
| 554 | .p = { .min = IRONLAKE_DAC_P_MIN, .max = IRONLAKE_DAC_P_MAX }, | 279 | .p = { .min = 5, .max = 80 }, |
| 555 | .p1 = { .min = IRONLAKE_DAC_P1_MIN, .max = IRONLAKE_DAC_P1_MAX }, | 280 | .p1 = { .min = 1, .max = 8 }, |
| 556 | .p2 = { .dot_limit = IRONLAKE_P2_DOT_LIMIT, | 281 | .p2 = { .dot_limit = 225000, |
| 557 | .p2_slow = IRONLAKE_DAC_P2_SLOW, | 282 | .p2_slow = 10, .p2_fast = 5 }, |
| 558 | .p2_fast = IRONLAKE_DAC_P2_FAST }, | ||
| 559 | .find_pll = intel_g4x_find_best_PLL, | 283 | .find_pll = intel_g4x_find_best_PLL, |
| 560 | }; | 284 | }; |
| 561 | 285 | ||
| 562 | static const intel_limit_t intel_limits_ironlake_single_lvds = { | 286 | static const intel_limit_t intel_limits_ironlake_single_lvds = { |
| 563 | .dot = { .min = IRONLAKE_DOT_MIN, .max = IRONLAKE_DOT_MAX }, | 287 | .dot = { .min = 25000, .max = 350000 }, |
| 564 | .vco = { .min = IRONLAKE_VCO_MIN, .max = IRONLAKE_VCO_MAX }, | 288 | .vco = { .min = 1760000, .max = 3510000 }, |
| 565 | .n = { .min = IRONLAKE_LVDS_S_N_MIN, .max = IRONLAKE_LVDS_S_N_MAX }, | 289 | .n = { .min = 1, .max = 3 }, |
| 566 | .m = { .min = IRONLAKE_LVDS_S_M_MIN, .max = IRONLAKE_LVDS_S_M_MAX }, | 290 | .m = { .min = 79, .max = 118 }, |
| 567 | .m1 = { .min = IRONLAKE_M1_MIN, .max = IRONLAKE_M1_MAX }, | 291 | .m1 = { .min = 12, .max = 22 }, |
| 568 | .m2 = { .min = IRONLAKE_M2_MIN, .max = IRONLAKE_M2_MAX }, | 292 | .m2 = { .min = 5, .max = 9 }, |
| 569 | .p = { .min = IRONLAKE_LVDS_S_P_MIN, .max = IRONLAKE_LVDS_S_P_MAX }, | 293 | .p = { .min = 28, .max = 112 }, |
| 570 | .p1 = { .min = IRONLAKE_LVDS_S_P1_MIN, .max = IRONLAKE_LVDS_S_P1_MAX }, | 294 | .p1 = { .min = 2, .max = 8 }, |
| 571 | .p2 = { .dot_limit = IRONLAKE_P2_DOT_LIMIT, | 295 | .p2 = { .dot_limit = 225000, |
| 572 | .p2_slow = IRONLAKE_LVDS_S_P2_SLOW, | 296 | .p2_slow = 14, .p2_fast = 14 }, |
| 573 | .p2_fast = IRONLAKE_LVDS_S_P2_FAST }, | ||
| 574 | .find_pll = intel_g4x_find_best_PLL, | 297 | .find_pll = intel_g4x_find_best_PLL, |
| 575 | }; | 298 | }; |
| 576 | 299 | ||
| 577 | static const intel_limit_t intel_limits_ironlake_dual_lvds = { | 300 | static const intel_limit_t intel_limits_ironlake_dual_lvds = { |
| 578 | .dot = { .min = IRONLAKE_DOT_MIN, .max = IRONLAKE_DOT_MAX }, | 301 | .dot = { .min = 25000, .max = 350000 }, |
| 579 | .vco = { .min = IRONLAKE_VCO_MIN, .max = IRONLAKE_VCO_MAX }, | 302 | .vco = { .min = 1760000, .max = 3510000 }, |
| 580 | .n = { .min = IRONLAKE_LVDS_D_N_MIN, .max = IRONLAKE_LVDS_D_N_MAX }, | 303 | .n = { .min = 1, .max = 3 }, |
| 581 | .m = { .min = IRONLAKE_LVDS_D_M_MIN, .max = IRONLAKE_LVDS_D_M_MAX }, | 304 | .m = { .min = 79, .max = 127 }, |
| 582 | .m1 = { .min = IRONLAKE_M1_MIN, .max = IRONLAKE_M1_MAX }, | 305 | .m1 = { .min = 12, .max = 22 }, |
| 583 | .m2 = { .min = IRONLAKE_M2_MIN, .max = IRONLAKE_M2_MAX }, | 306 | .m2 = { .min = 5, .max = 9 }, |
| 584 | .p = { .min = IRONLAKE_LVDS_D_P_MIN, .max = IRONLAKE_LVDS_D_P_MAX }, | 307 | .p = { .min = 14, .max = 56 }, |
| 585 | .p1 = { .min = IRONLAKE_LVDS_D_P1_MIN, .max = IRONLAKE_LVDS_D_P1_MAX }, | 308 | .p1 = { .min = 2, .max = 8 }, |
| 586 | .p2 = { .dot_limit = IRONLAKE_P2_DOT_LIMIT, | 309 | .p2 = { .dot_limit = 225000, |
| 587 | .p2_slow = IRONLAKE_LVDS_D_P2_SLOW, | 310 | .p2_slow = 7, .p2_fast = 7 }, |
| 588 | .p2_fast = IRONLAKE_LVDS_D_P2_FAST }, | ||
| 589 | .find_pll = intel_g4x_find_best_PLL, | 311 | .find_pll = intel_g4x_find_best_PLL, |
| 590 | }; | 312 | }; |
| 591 | 313 | ||
| 314 | /* LVDS 100mhz refclk limits. */ | ||
| 592 | static const intel_limit_t intel_limits_ironlake_single_lvds_100m = { | 315 | static const intel_limit_t intel_limits_ironlake_single_lvds_100m = { |
| 593 | .dot = { .min = IRONLAKE_DOT_MIN, .max = IRONLAKE_DOT_MAX }, | 316 | .dot = { .min = 25000, .max = 350000 }, |
| 594 | .vco = { .min = IRONLAKE_VCO_MIN, .max = IRONLAKE_VCO_MAX }, | 317 | .vco = { .min = 1760000, .max = 3510000 }, |
| 595 | .n = { .min = IRONLAKE_LVDS_S_SSC_N_MIN, .max = IRONLAKE_LVDS_S_SSC_N_MAX }, | 318 | .n = { .min = 1, .max = 2 }, |
| 596 | .m = { .min = IRONLAKE_LVDS_S_SSC_M_MIN, .max = IRONLAKE_LVDS_S_SSC_M_MAX }, | 319 | .m = { .min = 79, .max = 126 }, |
| 597 | .m1 = { .min = IRONLAKE_M1_MIN, .max = IRONLAKE_M1_MAX }, | 320 | .m1 = { .min = 12, .max = 22 }, |
| 598 | .m2 = { .min = IRONLAKE_M2_MIN, .max = IRONLAKE_M2_MAX }, | 321 | .m2 = { .min = 5, .max = 9 }, |
| 599 | .p = { .min = IRONLAKE_LVDS_S_SSC_P_MIN, .max = IRONLAKE_LVDS_S_SSC_P_MAX }, | 322 | .p = { .min = 28, .max = 112 }, |
| 600 | .p1 = { .min = IRONLAKE_LVDS_S_SSC_P1_MIN,.max = IRONLAKE_LVDS_S_SSC_P1_MAX }, | 323 | .p1 = { .min = 2,.max = 8 }, |
| 601 | .p2 = { .dot_limit = IRONLAKE_P2_DOT_LIMIT, | 324 | .p2 = { .dot_limit = 225000, |
| 602 | .p2_slow = IRONLAKE_LVDS_S_SSC_P2_SLOW, | 325 | .p2_slow = 14, .p2_fast = 14 }, |
| 603 | .p2_fast = IRONLAKE_LVDS_S_SSC_P2_FAST }, | ||
| 604 | .find_pll = intel_g4x_find_best_PLL, | 326 | .find_pll = intel_g4x_find_best_PLL, |
| 605 | }; | 327 | }; |
| 606 | 328 | ||
| 607 | static const intel_limit_t intel_limits_ironlake_dual_lvds_100m = { | 329 | static const intel_limit_t intel_limits_ironlake_dual_lvds_100m = { |
| 608 | .dot = { .min = IRONLAKE_DOT_MIN, .max = IRONLAKE_DOT_MAX }, | 330 | .dot = { .min = 25000, .max = 350000 }, |
| 609 | .vco = { .min = IRONLAKE_VCO_MIN, .max = IRONLAKE_VCO_MAX }, | 331 | .vco = { .min = 1760000, .max = 3510000 }, |
| 610 | .n = { .min = IRONLAKE_LVDS_D_SSC_N_MIN, .max = IRONLAKE_LVDS_D_SSC_N_MAX }, | 332 | .n = { .min = 1, .max = 3 }, |
| 611 | .m = { .min = IRONLAKE_LVDS_D_SSC_M_MIN, .max = IRONLAKE_LVDS_D_SSC_M_MAX }, | 333 | .m = { .min = 79, .max = 126 }, |
| 612 | .m1 = { .min = IRONLAKE_M1_MIN, .max = IRONLAKE_M1_MAX }, | 334 | .m1 = { .min = 12, .max = 22 }, |
| 613 | .m2 = { .min = IRONLAKE_M2_MIN, .max = IRONLAKE_M2_MAX }, | 335 | .m2 = { .min = 5, .max = 9 }, |
| 614 | .p = { .min = IRONLAKE_LVDS_D_SSC_P_MIN, .max = IRONLAKE_LVDS_D_SSC_P_MAX }, | 336 | .p = { .min = 14, .max = 42 }, |
| 615 | .p1 = { .min = IRONLAKE_LVDS_D_SSC_P1_MIN,.max = IRONLAKE_LVDS_D_SSC_P1_MAX }, | 337 | .p1 = { .min = 2,.max = 6 }, |
| 616 | .p2 = { .dot_limit = IRONLAKE_P2_DOT_LIMIT, | 338 | .p2 = { .dot_limit = 225000, |
| 617 | .p2_slow = IRONLAKE_LVDS_D_SSC_P2_SLOW, | 339 | .p2_slow = 7, .p2_fast = 7 }, |
| 618 | .p2_fast = IRONLAKE_LVDS_D_SSC_P2_FAST }, | ||
| 619 | .find_pll = intel_g4x_find_best_PLL, | 340 | .find_pll = intel_g4x_find_best_PLL, |
| 620 | }; | 341 | }; |
| 621 | 342 | ||
| 622 | static const intel_limit_t intel_limits_ironlake_display_port = { | 343 | static const intel_limit_t intel_limits_ironlake_display_port = { |
| 623 | .dot = { .min = IRONLAKE_DOT_MIN, | 344 | .dot = { .min = 25000, .max = 350000 }, |
| 624 | .max = IRONLAKE_DOT_MAX }, | 345 | .vco = { .min = 1760000, .max = 3510000}, |
| 625 | .vco = { .min = IRONLAKE_VCO_MIN, | 346 | .n = { .min = 1, .max = 2 }, |
| 626 | .max = IRONLAKE_VCO_MAX}, | 347 | .m = { .min = 81, .max = 90 }, |
| 627 | .n = { .min = IRONLAKE_DP_N_MIN, | 348 | .m1 = { .min = 12, .max = 22 }, |
| 628 | .max = IRONLAKE_DP_N_MAX }, | 349 | .m2 = { .min = 5, .max = 9 }, |
| 629 | .m = { .min = IRONLAKE_DP_M_MIN, | 350 | .p = { .min = 10, .max = 20 }, |
| 630 | .max = IRONLAKE_DP_M_MAX }, | 351 | .p1 = { .min = 1, .max = 2}, |
| 631 | .m1 = { .min = IRONLAKE_M1_MIN, | 352 | .p2 = { .dot_limit = 0, |
| 632 | .max = IRONLAKE_M1_MAX }, | 353 | .p2_slow = 10, .p2_fast = 10 }, |
| 633 | .m2 = { .min = IRONLAKE_M2_MIN, | ||
| 634 | .max = IRONLAKE_M2_MAX }, | ||
| 635 | .p = { .min = IRONLAKE_DP_P_MIN, | ||
| 636 | .max = IRONLAKE_DP_P_MAX }, | ||
| 637 | .p1 = { .min = IRONLAKE_DP_P1_MIN, | ||
| 638 | .max = IRONLAKE_DP_P1_MAX}, | ||
| 639 | .p2 = { .dot_limit = IRONLAKE_DP_P2_LIMIT, | ||
| 640 | .p2_slow = IRONLAKE_DP_P2_SLOW, | ||
| 641 | .p2_fast = IRONLAKE_DP_P2_FAST }, | ||
| 642 | .find_pll = intel_find_pll_ironlake_dp, | 354 | .find_pll = intel_find_pll_ironlake_dp, |
| 643 | }; | 355 | }; |
| 644 | 356 | ||
| @@ -1828,7 +1540,7 @@ static void sandybridge_blit_fbc_update(struct drm_device *dev) | |||
| 1828 | u32 blt_ecoskpd; | 1540 | u32 blt_ecoskpd; |
| 1829 | 1541 | ||
| 1830 | /* Make sure blitter notifies FBC of writes */ | 1542 | /* Make sure blitter notifies FBC of writes */ |
| 1831 | __gen6_gt_force_wake_get(dev_priv); | 1543 | gen6_gt_force_wake_get(dev_priv); |
| 1832 | blt_ecoskpd = I915_READ(GEN6_BLITTER_ECOSKPD); | 1544 | blt_ecoskpd = I915_READ(GEN6_BLITTER_ECOSKPD); |
| 1833 | blt_ecoskpd |= GEN6_BLITTER_FBC_NOTIFY << | 1545 | blt_ecoskpd |= GEN6_BLITTER_FBC_NOTIFY << |
| 1834 | GEN6_BLITTER_LOCK_SHIFT; | 1546 | GEN6_BLITTER_LOCK_SHIFT; |
| @@ -1839,7 +1551,7 @@ static void sandybridge_blit_fbc_update(struct drm_device *dev) | |||
| 1839 | GEN6_BLITTER_LOCK_SHIFT); | 1551 | GEN6_BLITTER_LOCK_SHIFT); |
| 1840 | I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); | 1552 | I915_WRITE(GEN6_BLITTER_ECOSKPD, blt_ecoskpd); |
| 1841 | POSTING_READ(GEN6_BLITTER_ECOSKPD); | 1553 | POSTING_READ(GEN6_BLITTER_ECOSKPD); |
| 1842 | __gen6_gt_force_wake_put(dev_priv); | 1554 | gen6_gt_force_wake_put(dev_priv); |
| 1843 | } | 1555 | } |
| 1844 | 1556 | ||
| 1845 | static void ironlake_enable_fbc(struct drm_crtc *crtc, unsigned long interval) | 1557 | static void ironlake_enable_fbc(struct drm_crtc *crtc, unsigned long interval) |
| @@ -2339,8 +2051,13 @@ static void intel_fdi_normal_train(struct drm_crtc *crtc) | |||
| 2339 | /* enable normal train */ | 2051 | /* enable normal train */ |
| 2340 | reg = FDI_TX_CTL(pipe); | 2052 | reg = FDI_TX_CTL(pipe); |
| 2341 | temp = I915_READ(reg); | 2053 | temp = I915_READ(reg); |
| 2342 | temp &= ~FDI_LINK_TRAIN_NONE; | 2054 | if (IS_GEN6(dev)) { |
| 2343 | temp |= FDI_LINK_TRAIN_NONE | FDI_TX_ENHANCE_FRAME_ENABLE; | 2055 | temp &= ~FDI_LINK_TRAIN_NONE; |
| 2056 | temp |= FDI_LINK_TRAIN_NONE | FDI_TX_ENHANCE_FRAME_ENABLE; | ||
| 2057 | } else if (IS_IVYBRIDGE(dev)) { | ||
| 2058 | temp &= ~FDI_LINK_TRAIN_NONE_IVB; | ||
| 2059 | temp |= FDI_LINK_TRAIN_NONE_IVB | FDI_TX_ENHANCE_FRAME_ENABLE; | ||
| 2060 | } | ||
| 2344 | I915_WRITE(reg, temp); | 2061 | I915_WRITE(reg, temp); |
| 2345 | 2062 | ||
| 2346 | reg = FDI_RX_CTL(pipe); | 2063 | reg = FDI_RX_CTL(pipe); |
| @@ -2357,6 +2074,11 @@ static void intel_fdi_normal_train(struct drm_crtc *crtc) | |||
| 2357 | /* wait one idle pattern time */ | 2074 | /* wait one idle pattern time */ |
| 2358 | POSTING_READ(reg); | 2075 | POSTING_READ(reg); |
| 2359 | udelay(1000); | 2076 | udelay(1000); |
| 2077 | |||
| 2078 | /* IVB wants error correction enabled */ | ||
| 2079 | if (IS_IVYBRIDGE(dev)) | ||
| 2080 | I915_WRITE(reg, I915_READ(reg) | FDI_FS_ERRC_ENABLE | | ||
| 2081 | FDI_FE_ERRC_ENABLE); | ||
| 2360 | } | 2082 | } |
| 2361 | 2083 | ||
| 2362 | /* The FDI link training functions for ILK/Ibexpeak. */ | 2084 | /* The FDI link training functions for ILK/Ibexpeak. */ |
| @@ -2584,7 +2306,116 @@ static void gen6_fdi_link_train(struct drm_crtc *crtc) | |||
| 2584 | DRM_DEBUG_KMS("FDI train done.\n"); | 2306 | DRM_DEBUG_KMS("FDI train done.\n"); |
| 2585 | } | 2307 | } |
| 2586 | 2308 | ||
| 2587 | static void ironlake_fdi_enable(struct drm_crtc *crtc) | 2309 | /* Manual link training for Ivy Bridge A0 parts */ |
| 2310 | static void ivb_manual_fdi_link_train(struct drm_crtc *crtc) | ||
| 2311 | { | ||
| 2312 | struct drm_device *dev = crtc->dev; | ||
| 2313 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 2314 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); | ||
| 2315 | int pipe = intel_crtc->pipe; | ||
| 2316 | u32 reg, temp, i; | ||
| 2317 | |||
| 2318 | /* Train 1: umask FDI RX Interrupt symbol_lock and bit_lock bit | ||
| 2319 | for train result */ | ||
| 2320 | reg = FDI_RX_IMR(pipe); | ||
| 2321 | temp = I915_READ(reg); | ||
| 2322 | temp &= ~FDI_RX_SYMBOL_LOCK; | ||
| 2323 | temp &= ~FDI_RX_BIT_LOCK; | ||
| 2324 | I915_WRITE(reg, temp); | ||
| 2325 | |||
| 2326 | POSTING_READ(reg); | ||
| 2327 | udelay(150); | ||
| 2328 | |||
| 2329 | /* enable CPU FDI TX and PCH FDI RX */ | ||
| 2330 | reg = FDI_TX_CTL(pipe); | ||
| 2331 | temp = I915_READ(reg); | ||
| 2332 | temp &= ~(7 << 19); | ||
| 2333 | temp |= (intel_crtc->fdi_lanes - 1) << 19; | ||
| 2334 | temp &= ~(FDI_LINK_TRAIN_AUTO | FDI_LINK_TRAIN_NONE_IVB); | ||
| 2335 | temp |= FDI_LINK_TRAIN_PATTERN_1_IVB; | ||
| 2336 | temp &= ~FDI_LINK_TRAIN_VOL_EMP_MASK; | ||
| 2337 | temp |= FDI_LINK_TRAIN_400MV_0DB_SNB_B; | ||
| 2338 | I915_WRITE(reg, temp | FDI_TX_ENABLE); | ||
| 2339 | |||
| 2340 | reg = FDI_RX_CTL(pipe); | ||
| 2341 | temp = I915_READ(reg); | ||
| 2342 | temp &= ~FDI_LINK_TRAIN_AUTO; | ||
| 2343 | temp &= ~FDI_LINK_TRAIN_PATTERN_MASK_CPT; | ||
| 2344 | temp |= FDI_LINK_TRAIN_PATTERN_1_CPT; | ||
| 2345 | I915_WRITE(reg, temp | FDI_RX_ENABLE); | ||
| 2346 | |||
| 2347 | POSTING_READ(reg); | ||
| 2348 | udelay(150); | ||
| 2349 | |||
| 2350 | for (i = 0; i < 4; i++ ) { | ||
| 2351 | reg = FDI_TX_CTL(pipe); | ||
| 2352 | temp = I915_READ(reg); | ||
| 2353 | temp &= ~FDI_LINK_TRAIN_VOL_EMP_MASK; | ||
| 2354 | temp |= snb_b_fdi_train_param[i]; | ||
| 2355 | I915_WRITE(reg, temp); | ||
| 2356 | |||
| 2357 | POSTING_READ(reg); | ||
| 2358 | udelay(500); | ||
| 2359 | |||
| 2360 | reg = FDI_RX_IIR(pipe); | ||
| 2361 | temp = I915_READ(reg); | ||
| 2362 | DRM_DEBUG_KMS("FDI_RX_IIR 0x%x\n", temp); | ||
| 2363 | |||
| 2364 | if (temp & FDI_RX_BIT_LOCK || | ||
| 2365 | (I915_READ(reg) & FDI_RX_BIT_LOCK)) { | ||
| 2366 | I915_WRITE(reg, temp | FDI_RX_BIT_LOCK); | ||
| 2367 | DRM_DEBUG_KMS("FDI train 1 done.\n"); | ||
| 2368 | break; | ||
| 2369 | } | ||
| 2370 | } | ||
| 2371 | if (i == 4) | ||
| 2372 | DRM_ERROR("FDI train 1 fail!\n"); | ||
| 2373 | |||
| 2374 | /* Train 2 */ | ||
| 2375 | reg = FDI_TX_CTL(pipe); | ||
| 2376 | temp = I915_READ(reg); | ||
| 2377 | temp &= ~FDI_LINK_TRAIN_NONE_IVB; | ||
| 2378 | temp |= FDI_LINK_TRAIN_PATTERN_2_IVB; | ||
| 2379 | temp &= ~FDI_LINK_TRAIN_VOL_EMP_MASK; | ||
| 2380 | temp |= FDI_LINK_TRAIN_400MV_0DB_SNB_B; | ||
| 2381 | I915_WRITE(reg, temp); | ||
| 2382 | |||
| 2383 | reg = FDI_RX_CTL(pipe); | ||
| 2384 | temp = I915_READ(reg); | ||
| 2385 | temp &= ~FDI_LINK_TRAIN_PATTERN_MASK_CPT; | ||
| 2386 | temp |= FDI_LINK_TRAIN_PATTERN_2_CPT; | ||
| 2387 | I915_WRITE(reg, temp); | ||
| 2388 | |||
| 2389 | POSTING_READ(reg); | ||
| 2390 | udelay(150); | ||
| 2391 | |||
| 2392 | for (i = 0; i < 4; i++ ) { | ||
| 2393 | reg = FDI_TX_CTL(pipe); | ||
| 2394 | temp = I915_READ(reg); | ||
| 2395 | temp &= ~FDI_LINK_TRAIN_VOL_EMP_MASK; | ||
| 2396 | temp |= snb_b_fdi_train_param[i]; | ||
| 2397 | I915_WRITE(reg, temp); | ||
| 2398 | |||
| 2399 | POSTING_READ(reg); | ||
| 2400 | udelay(500); | ||
| 2401 | |||
| 2402 | reg = FDI_RX_IIR(pipe); | ||
| 2403 | temp = I915_READ(reg); | ||
| 2404 | DRM_DEBUG_KMS("FDI_RX_IIR 0x%x\n", temp); | ||
| 2405 | |||
| 2406 | if (temp & FDI_RX_SYMBOL_LOCK) { | ||
| 2407 | I915_WRITE(reg, temp | FDI_RX_SYMBOL_LOCK); | ||
| 2408 | DRM_DEBUG_KMS("FDI train 2 done.\n"); | ||
| 2409 | break; | ||
| 2410 | } | ||
| 2411 | } | ||
| 2412 | if (i == 4) | ||
| 2413 | DRM_ERROR("FDI train 2 fail!\n"); | ||
| 2414 | |||
| 2415 | DRM_DEBUG_KMS("FDI train done.\n"); | ||
| 2416 | } | ||
| 2417 | |||
| 2418 | static void ironlake_fdi_pll_enable(struct drm_crtc *crtc) | ||
| 2588 | { | 2419 | { |
| 2589 | struct drm_device *dev = crtc->dev; | 2420 | struct drm_device *dev = crtc->dev; |
| 2590 | struct drm_i915_private *dev_priv = dev->dev_private; | 2421 | struct drm_i915_private *dev_priv = dev->dev_private; |
| @@ -2757,10 +2588,7 @@ static void ironlake_pch_enable(struct drm_crtc *crtc) | |||
| 2757 | u32 reg, temp; | 2588 | u32 reg, temp; |
| 2758 | 2589 | ||
| 2759 | /* For PCH output, training FDI link */ | 2590 | /* For PCH output, training FDI link */ |
| 2760 | if (IS_GEN6(dev)) | 2591 | dev_priv->display.fdi_link_train(crtc); |
| 2761 | gen6_fdi_link_train(crtc); | ||
| 2762 | else | ||
| 2763 | ironlake_fdi_link_train(crtc); | ||
| 2764 | 2592 | ||
| 2765 | intel_enable_pch_pll(dev_priv, pipe); | 2593 | intel_enable_pch_pll(dev_priv, pipe); |
| 2766 | 2594 | ||
| @@ -2850,7 +2678,7 @@ static void ironlake_crtc_enable(struct drm_crtc *crtc) | |||
| 2850 | is_pch_port = intel_crtc_driving_pch(crtc); | 2678 | is_pch_port = intel_crtc_driving_pch(crtc); |
| 2851 | 2679 | ||
| 2852 | if (is_pch_port) | 2680 | if (is_pch_port) |
| 2853 | ironlake_fdi_enable(crtc); | 2681 | ironlake_fdi_pll_enable(crtc); |
| 2854 | else | 2682 | else |
| 2855 | ironlake_fdi_disable(crtc); | 2683 | ironlake_fdi_disable(crtc); |
| 2856 | 2684 | ||
| @@ -2873,7 +2701,11 @@ static void ironlake_crtc_enable(struct drm_crtc *crtc) | |||
| 2873 | ironlake_pch_enable(crtc); | 2701 | ironlake_pch_enable(crtc); |
| 2874 | 2702 | ||
| 2875 | intel_crtc_load_lut(crtc); | 2703 | intel_crtc_load_lut(crtc); |
| 2704 | |||
| 2705 | mutex_lock(&dev->struct_mutex); | ||
| 2876 | intel_update_fbc(dev); | 2706 | intel_update_fbc(dev); |
| 2707 | mutex_unlock(&dev->struct_mutex); | ||
| 2708 | |||
| 2877 | intel_crtc_update_cursor(crtc, true); | 2709 | intel_crtc_update_cursor(crtc, true); |
| 2878 | } | 2710 | } |
| 2879 | 2711 | ||
| @@ -2969,8 +2801,11 @@ static void ironlake_crtc_disable(struct drm_crtc *crtc) | |||
| 2969 | 2801 | ||
| 2970 | intel_crtc->active = false; | 2802 | intel_crtc->active = false; |
| 2971 | intel_update_watermarks(dev); | 2803 | intel_update_watermarks(dev); |
| 2804 | |||
| 2805 | mutex_lock(&dev->struct_mutex); | ||
| 2972 | intel_update_fbc(dev); | 2806 | intel_update_fbc(dev); |
| 2973 | intel_clear_scanline_wait(dev); | 2807 | intel_clear_scanline_wait(dev); |
| 2808 | mutex_unlock(&dev->struct_mutex); | ||
| 2974 | } | 2809 | } |
| 2975 | 2810 | ||
| 2976 | static void ironlake_crtc_dpms(struct drm_crtc *crtc, int mode) | 2811 | static void ironlake_crtc_dpms(struct drm_crtc *crtc, int mode) |
| @@ -4516,34 +4351,28 @@ static inline bool intel_panel_use_ssc(struct drm_i915_private *dev_priv) | |||
| 4516 | return dev_priv->lvds_use_ssc && i915_panel_use_ssc; | 4351 | return dev_priv->lvds_use_ssc && i915_panel_use_ssc; |
| 4517 | } | 4352 | } |
| 4518 | 4353 | ||
| 4519 | static int intel_crtc_mode_set(struct drm_crtc *crtc, | 4354 | static int i9xx_crtc_mode_set(struct drm_crtc *crtc, |
| 4520 | struct drm_display_mode *mode, | 4355 | struct drm_display_mode *mode, |
| 4521 | struct drm_display_mode *adjusted_mode, | 4356 | struct drm_display_mode *adjusted_mode, |
| 4522 | int x, int y, | 4357 | int x, int y, |
| 4523 | struct drm_framebuffer *old_fb) | 4358 | struct drm_framebuffer *old_fb) |
| 4524 | { | 4359 | { |
| 4525 | struct drm_device *dev = crtc->dev; | 4360 | struct drm_device *dev = crtc->dev; |
| 4526 | struct drm_i915_private *dev_priv = dev->dev_private; | 4361 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 4527 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); | 4362 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); |
| 4528 | int pipe = intel_crtc->pipe; | 4363 | int pipe = intel_crtc->pipe; |
| 4529 | int plane = intel_crtc->plane; | 4364 | int plane = intel_crtc->plane; |
| 4530 | u32 fp_reg, dpll_reg; | ||
| 4531 | int refclk, num_connectors = 0; | 4365 | int refclk, num_connectors = 0; |
| 4532 | intel_clock_t clock, reduced_clock; | 4366 | intel_clock_t clock, reduced_clock; |
| 4533 | u32 dpll, fp = 0, fp2 = 0, dspcntr, pipeconf; | 4367 | u32 dpll, fp = 0, fp2 = 0, dspcntr, pipeconf; |
| 4534 | bool ok, has_reduced_clock = false, is_sdvo = false, is_dvo = false; | 4368 | bool ok, has_reduced_clock = false, is_sdvo = false, is_dvo = false; |
| 4535 | bool is_crt = false, is_lvds = false, is_tv = false, is_dp = false; | 4369 | bool is_crt = false, is_lvds = false, is_tv = false, is_dp = false; |
| 4536 | struct intel_encoder *has_edp_encoder = NULL; | ||
| 4537 | struct drm_mode_config *mode_config = &dev->mode_config; | 4370 | struct drm_mode_config *mode_config = &dev->mode_config; |
| 4538 | struct intel_encoder *encoder; | 4371 | struct intel_encoder *encoder; |
| 4539 | const intel_limit_t *limit; | 4372 | const intel_limit_t *limit; |
| 4540 | int ret; | 4373 | int ret; |
| 4541 | struct fdi_m_n m_n = {0}; | 4374 | u32 temp; |
| 4542 | u32 reg, temp; | ||
| 4543 | u32 lvds_sync = 0; | 4375 | u32 lvds_sync = 0; |
| 4544 | int target_clock; | ||
| 4545 | |||
| 4546 | drm_vblank_pre_modeset(dev, pipe); | ||
| 4547 | 4376 | ||
| 4548 | list_for_each_entry(encoder, &mode_config->encoder_list, base.head) { | 4377 | list_for_each_entry(encoder, &mode_config->encoder_list, base.head) { |
| 4549 | if (encoder->base.crtc != crtc) | 4378 | if (encoder->base.crtc != crtc) |
| @@ -4571,9 +4400,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4571 | case INTEL_OUTPUT_DISPLAYPORT: | 4400 | case INTEL_OUTPUT_DISPLAYPORT: |
| 4572 | is_dp = true; | 4401 | is_dp = true; |
| 4573 | break; | 4402 | break; |
| 4574 | case INTEL_OUTPUT_EDP: | ||
| 4575 | has_edp_encoder = encoder; | ||
| 4576 | break; | ||
| 4577 | } | 4403 | } |
| 4578 | 4404 | ||
| 4579 | num_connectors++; | 4405 | num_connectors++; |
| @@ -4585,9 +4411,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4585 | refclk / 1000); | 4411 | refclk / 1000); |
| 4586 | } else if (!IS_GEN2(dev)) { | 4412 | } else if (!IS_GEN2(dev)) { |
| 4587 | refclk = 96000; | 4413 | refclk = 96000; |
| 4588 | if (HAS_PCH_SPLIT(dev) && | ||
| 4589 | (!has_edp_encoder || intel_encoder_is_pch_edp(&has_edp_encoder->base))) | ||
| 4590 | refclk = 120000; /* 120Mhz refclk */ | ||
| 4591 | } else { | 4414 | } else { |
| 4592 | refclk = 48000; | 4415 | refclk = 48000; |
| 4593 | } | 4416 | } |
| @@ -4601,7 +4424,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4601 | ok = limit->find_pll(limit, crtc, adjusted_mode->clock, refclk, &clock); | 4424 | ok = limit->find_pll(limit, crtc, adjusted_mode->clock, refclk, &clock); |
| 4602 | if (!ok) { | 4425 | if (!ok) { |
| 4603 | DRM_ERROR("Couldn't find PLL settings for mode!\n"); | 4426 | DRM_ERROR("Couldn't find PLL settings for mode!\n"); |
| 4604 | drm_vblank_post_modeset(dev, pipe); | ||
| 4605 | return -EINVAL; | 4427 | return -EINVAL; |
| 4606 | } | 4428 | } |
| 4607 | 4429 | ||
| @@ -4645,143 +4467,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4645 | } | 4467 | } |
| 4646 | } | 4468 | } |
| 4647 | 4469 | ||
| 4648 | /* FDI link */ | ||
| 4649 | if (HAS_PCH_SPLIT(dev)) { | ||
| 4650 | int pixel_multiplier = intel_mode_get_pixel_multiplier(adjusted_mode); | ||
| 4651 | int lane = 0, link_bw, bpp; | ||
| 4652 | /* CPU eDP doesn't require FDI link, so just set DP M/N | ||
| 4653 | according to current link config */ | ||
| 4654 | if (has_edp_encoder && !intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | ||
| 4655 | target_clock = mode->clock; | ||
| 4656 | intel_edp_link_config(has_edp_encoder, | ||
| 4657 | &lane, &link_bw); | ||
| 4658 | } else { | ||
| 4659 | /* [e]DP over FDI requires target mode clock | ||
| 4660 | instead of link clock */ | ||
| 4661 | if (is_dp || intel_encoder_is_pch_edp(&has_edp_encoder->base)) | ||
| 4662 | target_clock = mode->clock; | ||
| 4663 | else | ||
| 4664 | target_clock = adjusted_mode->clock; | ||
| 4665 | |||
| 4666 | /* FDI is a binary signal running at ~2.7GHz, encoding | ||
| 4667 | * each output octet as 10 bits. The actual frequency | ||
| 4668 | * is stored as a divider into a 100MHz clock, and the | ||
| 4669 | * mode pixel clock is stored in units of 1KHz. | ||
| 4670 | * Hence the bw of each lane in terms of the mode signal | ||
| 4671 | * is: | ||
| 4672 | */ | ||
| 4673 | link_bw = intel_fdi_link_freq(dev) * MHz(100)/KHz(1)/10; | ||
| 4674 | } | ||
| 4675 | |||
| 4676 | /* determine panel color depth */ | ||
| 4677 | temp = I915_READ(PIPECONF(pipe)); | ||
| 4678 | temp &= ~PIPE_BPC_MASK; | ||
| 4679 | if (is_lvds) { | ||
| 4680 | /* the BPC will be 6 if it is 18-bit LVDS panel */ | ||
| 4681 | if ((I915_READ(PCH_LVDS) & LVDS_A3_POWER_MASK) == LVDS_A3_POWER_UP) | ||
| 4682 | temp |= PIPE_8BPC; | ||
| 4683 | else | ||
| 4684 | temp |= PIPE_6BPC; | ||
| 4685 | } else if (has_edp_encoder) { | ||
| 4686 | switch (dev_priv->edp.bpp/3) { | ||
| 4687 | case 8: | ||
| 4688 | temp |= PIPE_8BPC; | ||
| 4689 | break; | ||
| 4690 | case 10: | ||
| 4691 | temp |= PIPE_10BPC; | ||
| 4692 | break; | ||
| 4693 | case 6: | ||
| 4694 | temp |= PIPE_6BPC; | ||
| 4695 | break; | ||
| 4696 | case 12: | ||
| 4697 | temp |= PIPE_12BPC; | ||
| 4698 | break; | ||
| 4699 | } | ||
| 4700 | } else | ||
| 4701 | temp |= PIPE_8BPC; | ||
| 4702 | I915_WRITE(PIPECONF(pipe), temp); | ||
| 4703 | |||
| 4704 | switch (temp & PIPE_BPC_MASK) { | ||
| 4705 | case PIPE_8BPC: | ||
| 4706 | bpp = 24; | ||
| 4707 | break; | ||
| 4708 | case PIPE_10BPC: | ||
| 4709 | bpp = 30; | ||
| 4710 | break; | ||
| 4711 | case PIPE_6BPC: | ||
| 4712 | bpp = 18; | ||
| 4713 | break; | ||
| 4714 | case PIPE_12BPC: | ||
| 4715 | bpp = 36; | ||
| 4716 | break; | ||
| 4717 | default: | ||
| 4718 | DRM_ERROR("unknown pipe bpc value\n"); | ||
| 4719 | bpp = 24; | ||
| 4720 | } | ||
| 4721 | |||
| 4722 | if (!lane) { | ||
| 4723 | /* | ||
| 4724 | * Account for spread spectrum to avoid | ||
| 4725 | * oversubscribing the link. Max center spread | ||
| 4726 | * is 2.5%; use 5% for safety's sake. | ||
| 4727 | */ | ||
| 4728 | u32 bps = target_clock * bpp * 21 / 20; | ||
| 4729 | lane = bps / (link_bw * 8) + 1; | ||
| 4730 | } | ||
| 4731 | |||
| 4732 | intel_crtc->fdi_lanes = lane; | ||
| 4733 | |||
| 4734 | if (pixel_multiplier > 1) | ||
| 4735 | link_bw *= pixel_multiplier; | ||
| 4736 | ironlake_compute_m_n(bpp, lane, target_clock, link_bw, &m_n); | ||
| 4737 | } | ||
| 4738 | |||
| 4739 | /* Ironlake: try to setup display ref clock before DPLL | ||
| 4740 | * enabling. This is only under driver's control after | ||
| 4741 | * PCH B stepping, previous chipset stepping should be | ||
| 4742 | * ignoring this setting. | ||
| 4743 | */ | ||
| 4744 | if (HAS_PCH_SPLIT(dev)) { | ||
| 4745 | temp = I915_READ(PCH_DREF_CONTROL); | ||
| 4746 | /* Always enable nonspread source */ | ||
| 4747 | temp &= ~DREF_NONSPREAD_SOURCE_MASK; | ||
| 4748 | temp |= DREF_NONSPREAD_SOURCE_ENABLE; | ||
| 4749 | temp &= ~DREF_SSC_SOURCE_MASK; | ||
| 4750 | temp |= DREF_SSC_SOURCE_ENABLE; | ||
| 4751 | I915_WRITE(PCH_DREF_CONTROL, temp); | ||
| 4752 | |||
| 4753 | POSTING_READ(PCH_DREF_CONTROL); | ||
| 4754 | udelay(200); | ||
| 4755 | |||
| 4756 | if (has_edp_encoder) { | ||
| 4757 | if (intel_panel_use_ssc(dev_priv)) { | ||
| 4758 | temp |= DREF_SSC1_ENABLE; | ||
| 4759 | I915_WRITE(PCH_DREF_CONTROL, temp); | ||
| 4760 | |||
| 4761 | POSTING_READ(PCH_DREF_CONTROL); | ||
| 4762 | udelay(200); | ||
| 4763 | } | ||
| 4764 | temp &= ~DREF_CPU_SOURCE_OUTPUT_MASK; | ||
| 4765 | |||
| 4766 | /* Enable CPU source on CPU attached eDP */ | ||
| 4767 | if (!intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | ||
| 4768 | if (intel_panel_use_ssc(dev_priv)) | ||
| 4769 | temp |= DREF_CPU_SOURCE_OUTPUT_DOWNSPREAD; | ||
| 4770 | else | ||
| 4771 | temp |= DREF_CPU_SOURCE_OUTPUT_NONSPREAD; | ||
| 4772 | } else { | ||
| 4773 | /* Enable SSC on PCH eDP if needed */ | ||
| 4774 | if (intel_panel_use_ssc(dev_priv)) { | ||
| 4775 | DRM_ERROR("enabling SSC on PCH\n"); | ||
| 4776 | temp |= DREF_SUPERSPREAD_SOURCE_ENABLE; | ||
| 4777 | } | ||
| 4778 | } | ||
| 4779 | I915_WRITE(PCH_DREF_CONTROL, temp); | ||
| 4780 | POSTING_READ(PCH_DREF_CONTROL); | ||
| 4781 | udelay(200); | ||
| 4782 | } | ||
| 4783 | } | ||
| 4784 | |||
| 4785 | if (IS_PINEVIEW(dev)) { | 4470 | if (IS_PINEVIEW(dev)) { |
| 4786 | fp = (1 << clock.n) << 16 | clock.m1 << 8 | clock.m2; | 4471 | fp = (1 << clock.n) << 16 | clock.m1 << 8 | clock.m2; |
| 4787 | if (has_reduced_clock) | 4472 | if (has_reduced_clock) |
| @@ -4794,25 +4479,7 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4794 | reduced_clock.m2; | 4479 | reduced_clock.m2; |
| 4795 | } | 4480 | } |
| 4796 | 4481 | ||
| 4797 | /* Enable autotuning of the PLL clock (if permissible) */ | 4482 | dpll = DPLL_VGA_MODE_DIS; |
| 4798 | if (HAS_PCH_SPLIT(dev)) { | ||
| 4799 | int factor = 21; | ||
| 4800 | |||
| 4801 | if (is_lvds) { | ||
| 4802 | if ((intel_panel_use_ssc(dev_priv) && | ||
| 4803 | dev_priv->lvds_ssc_freq == 100) || | ||
| 4804 | (I915_READ(PCH_LVDS) & LVDS_CLKB_POWER_MASK) == LVDS_CLKB_POWER_UP) | ||
| 4805 | factor = 25; | ||
| 4806 | } else if (is_sdvo && is_tv) | ||
| 4807 | factor = 20; | ||
| 4808 | |||
| 4809 | if (clock.m1 < factor * clock.n) | ||
| 4810 | fp |= FP_CB_TUNE; | ||
| 4811 | } | ||
| 4812 | |||
| 4813 | dpll = 0; | ||
| 4814 | if (!HAS_PCH_SPLIT(dev)) | ||
| 4815 | dpll = DPLL_VGA_MODE_DIS; | ||
| 4816 | 4483 | ||
| 4817 | if (!IS_GEN2(dev)) { | 4484 | if (!IS_GEN2(dev)) { |
| 4818 | if (is_lvds) | 4485 | if (is_lvds) |
| @@ -4824,12 +4491,10 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4824 | if (pixel_multiplier > 1) { | 4491 | if (pixel_multiplier > 1) { |
| 4825 | if (IS_I945G(dev) || IS_I945GM(dev) || IS_G33(dev)) | 4492 | if (IS_I945G(dev) || IS_I945GM(dev) || IS_G33(dev)) |
| 4826 | dpll |= (pixel_multiplier - 1) << SDVO_MULTIPLIER_SHIFT_HIRES; | 4493 | dpll |= (pixel_multiplier - 1) << SDVO_MULTIPLIER_SHIFT_HIRES; |
| 4827 | else if (HAS_PCH_SPLIT(dev)) | ||
| 4828 | dpll |= (pixel_multiplier - 1) << PLL_REF_SDVO_HDMI_MULTIPLIER_SHIFT; | ||
| 4829 | } | 4494 | } |
| 4830 | dpll |= DPLL_DVO_HIGH_SPEED; | 4495 | dpll |= DPLL_DVO_HIGH_SPEED; |
| 4831 | } | 4496 | } |
| 4832 | if (is_dp || intel_encoder_is_pch_edp(&has_edp_encoder->base)) | 4497 | if (is_dp) |
| 4833 | dpll |= DPLL_DVO_HIGH_SPEED; | 4498 | dpll |= DPLL_DVO_HIGH_SPEED; |
| 4834 | 4499 | ||
| 4835 | /* compute bitmask from p1 value */ | 4500 | /* compute bitmask from p1 value */ |
| @@ -4837,9 +4502,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4837 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA01_P1_POST_DIV_SHIFT_PINEVIEW; | 4502 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA01_P1_POST_DIV_SHIFT_PINEVIEW; |
| 4838 | else { | 4503 | else { |
| 4839 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA01_P1_POST_DIV_SHIFT; | 4504 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA01_P1_POST_DIV_SHIFT; |
| 4840 | /* also FPA1 */ | ||
| 4841 | if (HAS_PCH_SPLIT(dev)) | ||
| 4842 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA1_P1_POST_DIV_SHIFT; | ||
| 4843 | if (IS_G4X(dev) && has_reduced_clock) | 4505 | if (IS_G4X(dev) && has_reduced_clock) |
| 4844 | dpll |= (1 << (reduced_clock.p1 - 1)) << DPLL_FPA1_P1_POST_DIV_SHIFT; | 4506 | dpll |= (1 << (reduced_clock.p1 - 1)) << DPLL_FPA1_P1_POST_DIV_SHIFT; |
| 4845 | } | 4507 | } |
| @@ -4857,7 +4519,7 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4857 | dpll |= DPLLB_LVDS_P2_CLOCK_DIV_14; | 4519 | dpll |= DPLLB_LVDS_P2_CLOCK_DIV_14; |
| 4858 | break; | 4520 | break; |
| 4859 | } | 4521 | } |
| 4860 | if (INTEL_INFO(dev)->gen >= 4 && !HAS_PCH_SPLIT(dev)) | 4522 | if (INTEL_INFO(dev)->gen >= 4) |
| 4861 | dpll |= (6 << PLL_LOAD_PULSE_PHASE_SHIFT); | 4523 | dpll |= (6 << PLL_LOAD_PULSE_PHASE_SHIFT); |
| 4862 | } else { | 4524 | } else { |
| 4863 | if (is_lvds) { | 4525 | if (is_lvds) { |
| @@ -4891,12 +4553,10 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4891 | 4553 | ||
| 4892 | /* Ironlake's plane is forced to pipe, bit 24 is to | 4554 | /* Ironlake's plane is forced to pipe, bit 24 is to |
| 4893 | enable color space conversion */ | 4555 | enable color space conversion */ |
| 4894 | if (!HAS_PCH_SPLIT(dev)) { | 4556 | if (pipe == 0) |
| 4895 | if (pipe == 0) | 4557 | dspcntr &= ~DISPPLANE_SEL_PIPE_MASK; |
| 4896 | dspcntr &= ~DISPPLANE_SEL_PIPE_MASK; | 4558 | else |
| 4897 | else | 4559 | dspcntr |= DISPPLANE_SEL_PIPE_B; |
| 4898 | dspcntr |= DISPPLANE_SEL_PIPE_B; | ||
| 4899 | } | ||
| 4900 | 4560 | ||
| 4901 | if (pipe == 0 && INTEL_INFO(dev)->gen < 4) { | 4561 | if (pipe == 0 && INTEL_INFO(dev)->gen < 4) { |
| 4902 | /* Enable pixel doubling when the dot clock is > 90% of the (display) | 4562 | /* Enable pixel doubling when the dot clock is > 90% of the (display) |
| @@ -4912,27 +4572,506 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4912 | pipeconf &= ~PIPECONF_DOUBLE_WIDE; | 4572 | pipeconf &= ~PIPECONF_DOUBLE_WIDE; |
| 4913 | } | 4573 | } |
| 4914 | 4574 | ||
| 4915 | if (!HAS_PCH_SPLIT(dev)) | 4575 | dpll |= DPLL_VCO_ENABLE; |
| 4916 | dpll |= DPLL_VCO_ENABLE; | ||
| 4917 | 4576 | ||
| 4918 | DRM_DEBUG_KMS("Mode for pipe %c:\n", pipe == 0 ? 'A' : 'B'); | 4577 | DRM_DEBUG_KMS("Mode for pipe %c:\n", pipe == 0 ? 'A' : 'B'); |
| 4919 | drm_mode_debug_printmodeline(mode); | 4578 | drm_mode_debug_printmodeline(mode); |
| 4920 | 4579 | ||
| 4921 | /* assign to Ironlake registers */ | 4580 | I915_WRITE(FP0(pipe), fp); |
| 4922 | if (HAS_PCH_SPLIT(dev)) { | 4581 | I915_WRITE(DPLL(pipe), dpll & ~DPLL_VCO_ENABLE); |
| 4923 | fp_reg = PCH_FP0(pipe); | 4582 | |
| 4924 | dpll_reg = PCH_DPLL(pipe); | 4583 | POSTING_READ(DPLL(pipe)); |
| 4584 | udelay(150); | ||
| 4585 | |||
| 4586 | /* The LVDS pin pair needs to be on before the DPLLs are enabled. | ||
| 4587 | * This is an exception to the general rule that mode_set doesn't turn | ||
| 4588 | * things on. | ||
| 4589 | */ | ||
| 4590 | if (is_lvds) { | ||
| 4591 | temp = I915_READ(LVDS); | ||
| 4592 | temp |= LVDS_PORT_EN | LVDS_A0A2_CLKA_POWER_UP; | ||
| 4593 | if (pipe == 1) { | ||
| 4594 | temp |= LVDS_PIPEB_SELECT; | ||
| 4595 | } else { | ||
| 4596 | temp &= ~LVDS_PIPEB_SELECT; | ||
| 4597 | } | ||
| 4598 | /* set the corresponsding LVDS_BORDER bit */ | ||
| 4599 | temp |= dev_priv->lvds_border_bits; | ||
| 4600 | /* Set the B0-B3 data pairs corresponding to whether we're going to | ||
| 4601 | * set the DPLLs for dual-channel mode or not. | ||
| 4602 | */ | ||
| 4603 | if (clock.p2 == 7) | ||
| 4604 | temp |= LVDS_B0B3_POWER_UP | LVDS_CLKB_POWER_UP; | ||
| 4605 | else | ||
| 4606 | temp &= ~(LVDS_B0B3_POWER_UP | LVDS_CLKB_POWER_UP); | ||
| 4607 | |||
| 4608 | /* It would be nice to set 24 vs 18-bit mode (LVDS_A3_POWER_UP) | ||
| 4609 | * appropriately here, but we need to look more thoroughly into how | ||
| 4610 | * panels behave in the two modes. | ||
| 4611 | */ | ||
| 4612 | /* set the dithering flag on LVDS as needed */ | ||
| 4613 | if (INTEL_INFO(dev)->gen >= 4) { | ||
| 4614 | if (dev_priv->lvds_dither) | ||
| 4615 | temp |= LVDS_ENABLE_DITHER; | ||
| 4616 | else | ||
| 4617 | temp &= ~LVDS_ENABLE_DITHER; | ||
| 4618 | } | ||
| 4619 | if (adjusted_mode->flags & DRM_MODE_FLAG_NHSYNC) | ||
| 4620 | lvds_sync |= LVDS_HSYNC_POLARITY; | ||
| 4621 | if (adjusted_mode->flags & DRM_MODE_FLAG_NVSYNC) | ||
| 4622 | lvds_sync |= LVDS_VSYNC_POLARITY; | ||
| 4623 | if ((temp & (LVDS_HSYNC_POLARITY | LVDS_VSYNC_POLARITY)) | ||
| 4624 | != lvds_sync) { | ||
| 4625 | char flags[2] = "-+"; | ||
| 4626 | DRM_INFO("Changing LVDS panel from " | ||
| 4627 | "(%chsync, %cvsync) to (%chsync, %cvsync)\n", | ||
| 4628 | flags[!(temp & LVDS_HSYNC_POLARITY)], | ||
| 4629 | flags[!(temp & LVDS_VSYNC_POLARITY)], | ||
| 4630 | flags[!(lvds_sync & LVDS_HSYNC_POLARITY)], | ||
| 4631 | flags[!(lvds_sync & LVDS_VSYNC_POLARITY)]); | ||
| 4632 | temp &= ~(LVDS_HSYNC_POLARITY | LVDS_VSYNC_POLARITY); | ||
| 4633 | temp |= lvds_sync; | ||
| 4634 | } | ||
| 4635 | I915_WRITE(LVDS, temp); | ||
| 4636 | } | ||
| 4637 | |||
| 4638 | if (is_dp) { | ||
| 4639 | intel_dp_set_m_n(crtc, mode, adjusted_mode); | ||
| 4640 | } | ||
| 4641 | |||
| 4642 | I915_WRITE(DPLL(pipe), dpll); | ||
| 4643 | |||
| 4644 | /* Wait for the clocks to stabilize. */ | ||
| 4645 | POSTING_READ(DPLL(pipe)); | ||
| 4646 | udelay(150); | ||
| 4647 | |||
| 4648 | if (INTEL_INFO(dev)->gen >= 4) { | ||
| 4649 | temp = 0; | ||
| 4650 | if (is_sdvo) { | ||
| 4651 | temp = intel_mode_get_pixel_multiplier(adjusted_mode); | ||
| 4652 | if (temp > 1) | ||
| 4653 | temp = (temp - 1) << DPLL_MD_UDI_MULTIPLIER_SHIFT; | ||
| 4654 | else | ||
| 4655 | temp = 0; | ||
| 4656 | } | ||
| 4657 | I915_WRITE(DPLL_MD(pipe), temp); | ||
| 4925 | } else { | 4658 | } else { |
| 4926 | fp_reg = FP0(pipe); | 4659 | /* The pixel multiplier can only be updated once the |
| 4927 | dpll_reg = DPLL(pipe); | 4660 | * DPLL is enabled and the clocks are stable. |
| 4661 | * | ||
| 4662 | * So write it again. | ||
| 4663 | */ | ||
| 4664 | I915_WRITE(DPLL(pipe), dpll); | ||
| 4665 | } | ||
| 4666 | |||
| 4667 | intel_crtc->lowfreq_avail = false; | ||
| 4668 | if (is_lvds && has_reduced_clock && i915_powersave) { | ||
| 4669 | I915_WRITE(FP1(pipe), fp2); | ||
| 4670 | intel_crtc->lowfreq_avail = true; | ||
| 4671 | if (HAS_PIPE_CXSR(dev)) { | ||
| 4672 | DRM_DEBUG_KMS("enabling CxSR downclocking\n"); | ||
| 4673 | pipeconf |= PIPECONF_CXSR_DOWNCLOCK; | ||
| 4674 | } | ||
| 4675 | } else { | ||
| 4676 | I915_WRITE(FP1(pipe), fp); | ||
| 4677 | if (HAS_PIPE_CXSR(dev)) { | ||
| 4678 | DRM_DEBUG_KMS("disabling CxSR downclocking\n"); | ||
| 4679 | pipeconf &= ~PIPECONF_CXSR_DOWNCLOCK; | ||
| 4680 | } | ||
| 4928 | } | 4681 | } |
| 4929 | 4682 | ||
| 4683 | if (adjusted_mode->flags & DRM_MODE_FLAG_INTERLACE) { | ||
| 4684 | pipeconf |= PIPECONF_INTERLACE_W_FIELD_INDICATION; | ||
| 4685 | /* the chip adds 2 halflines automatically */ | ||
| 4686 | adjusted_mode->crtc_vdisplay -= 1; | ||
| 4687 | adjusted_mode->crtc_vtotal -= 1; | ||
| 4688 | adjusted_mode->crtc_vblank_start -= 1; | ||
| 4689 | adjusted_mode->crtc_vblank_end -= 1; | ||
| 4690 | adjusted_mode->crtc_vsync_end -= 1; | ||
| 4691 | adjusted_mode->crtc_vsync_start -= 1; | ||
| 4692 | } else | ||
| 4693 | pipeconf &= ~PIPECONF_INTERLACE_W_FIELD_INDICATION; /* progressive */ | ||
| 4694 | |||
| 4695 | I915_WRITE(HTOTAL(pipe), | ||
| 4696 | (adjusted_mode->crtc_hdisplay - 1) | | ||
| 4697 | ((adjusted_mode->crtc_htotal - 1) << 16)); | ||
| 4698 | I915_WRITE(HBLANK(pipe), | ||
| 4699 | (adjusted_mode->crtc_hblank_start - 1) | | ||
| 4700 | ((adjusted_mode->crtc_hblank_end - 1) << 16)); | ||
| 4701 | I915_WRITE(HSYNC(pipe), | ||
| 4702 | (adjusted_mode->crtc_hsync_start - 1) | | ||
| 4703 | ((adjusted_mode->crtc_hsync_end - 1) << 16)); | ||
| 4704 | |||
| 4705 | I915_WRITE(VTOTAL(pipe), | ||
| 4706 | (adjusted_mode->crtc_vdisplay - 1) | | ||
| 4707 | ((adjusted_mode->crtc_vtotal - 1) << 16)); | ||
| 4708 | I915_WRITE(VBLANK(pipe), | ||
| 4709 | (adjusted_mode->crtc_vblank_start - 1) | | ||
| 4710 | ((adjusted_mode->crtc_vblank_end - 1) << 16)); | ||
| 4711 | I915_WRITE(VSYNC(pipe), | ||
| 4712 | (adjusted_mode->crtc_vsync_start - 1) | | ||
| 4713 | ((adjusted_mode->crtc_vsync_end - 1) << 16)); | ||
| 4714 | |||
| 4715 | /* pipesrc and dspsize control the size that is scaled from, | ||
| 4716 | * which should always be the user's requested size. | ||
| 4717 | */ | ||
| 4718 | I915_WRITE(DSPSIZE(plane), | ||
| 4719 | ((mode->vdisplay - 1) << 16) | | ||
| 4720 | (mode->hdisplay - 1)); | ||
| 4721 | I915_WRITE(DSPPOS(plane), 0); | ||
| 4722 | I915_WRITE(PIPESRC(pipe), | ||
| 4723 | ((mode->hdisplay - 1) << 16) | (mode->vdisplay - 1)); | ||
| 4724 | |||
| 4725 | I915_WRITE(PIPECONF(pipe), pipeconf); | ||
| 4726 | POSTING_READ(PIPECONF(pipe)); | ||
| 4727 | intel_enable_pipe(dev_priv, pipe, false); | ||
| 4728 | |||
| 4729 | intel_wait_for_vblank(dev, pipe); | ||
| 4730 | |||
| 4731 | I915_WRITE(DSPCNTR(plane), dspcntr); | ||
| 4732 | POSTING_READ(DSPCNTR(plane)); | ||
| 4733 | |||
| 4734 | ret = intel_pipe_set_base(crtc, x, y, old_fb); | ||
| 4735 | |||
| 4736 | intel_update_watermarks(dev); | ||
| 4737 | |||
| 4738 | return ret; | ||
| 4739 | } | ||
| 4740 | |||
| 4741 | static int ironlake_crtc_mode_set(struct drm_crtc *crtc, | ||
| 4742 | struct drm_display_mode *mode, | ||
| 4743 | struct drm_display_mode *adjusted_mode, | ||
| 4744 | int x, int y, | ||
| 4745 | struct drm_framebuffer *old_fb) | ||
| 4746 | { | ||
| 4747 | struct drm_device *dev = crtc->dev; | ||
| 4748 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 4749 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); | ||
| 4750 | int pipe = intel_crtc->pipe; | ||
| 4751 | int plane = intel_crtc->plane; | ||
| 4752 | int refclk, num_connectors = 0; | ||
| 4753 | intel_clock_t clock, reduced_clock; | ||
| 4754 | u32 dpll, fp = 0, fp2 = 0, dspcntr, pipeconf; | ||
| 4755 | bool ok, has_reduced_clock = false, is_sdvo = false; | ||
| 4756 | bool is_crt = false, is_lvds = false, is_tv = false, is_dp = false; | ||
| 4757 | struct intel_encoder *has_edp_encoder = NULL; | ||
| 4758 | struct drm_mode_config *mode_config = &dev->mode_config; | ||
| 4759 | struct intel_encoder *encoder; | ||
| 4760 | const intel_limit_t *limit; | ||
| 4761 | int ret; | ||
| 4762 | struct fdi_m_n m_n = {0}; | ||
| 4763 | u32 temp; | ||
| 4764 | u32 lvds_sync = 0; | ||
| 4765 | int target_clock, pixel_multiplier, lane, link_bw, bpp, factor; | ||
| 4766 | |||
| 4767 | list_for_each_entry(encoder, &mode_config->encoder_list, base.head) { | ||
| 4768 | if (encoder->base.crtc != crtc) | ||
| 4769 | continue; | ||
| 4770 | |||
| 4771 | switch (encoder->type) { | ||
| 4772 | case INTEL_OUTPUT_LVDS: | ||
| 4773 | is_lvds = true; | ||
| 4774 | break; | ||
| 4775 | case INTEL_OUTPUT_SDVO: | ||
| 4776 | case INTEL_OUTPUT_HDMI: | ||
| 4777 | is_sdvo = true; | ||
| 4778 | if (encoder->needs_tv_clock) | ||
| 4779 | is_tv = true; | ||
| 4780 | break; | ||
| 4781 | case INTEL_OUTPUT_TVOUT: | ||
| 4782 | is_tv = true; | ||
| 4783 | break; | ||
| 4784 | case INTEL_OUTPUT_ANALOG: | ||
| 4785 | is_crt = true; | ||
| 4786 | break; | ||
| 4787 | case INTEL_OUTPUT_DISPLAYPORT: | ||
| 4788 | is_dp = true; | ||
| 4789 | break; | ||
| 4790 | case INTEL_OUTPUT_EDP: | ||
| 4791 | has_edp_encoder = encoder; | ||
| 4792 | break; | ||
| 4793 | } | ||
| 4794 | |||
| 4795 | num_connectors++; | ||
| 4796 | } | ||
| 4797 | |||
| 4798 | if (is_lvds && intel_panel_use_ssc(dev_priv) && num_connectors < 2) { | ||
| 4799 | refclk = dev_priv->lvds_ssc_freq * 1000; | ||
| 4800 | DRM_DEBUG_KMS("using SSC reference clock of %d MHz\n", | ||
| 4801 | refclk / 1000); | ||
| 4802 | } else { | ||
| 4803 | refclk = 96000; | ||
| 4804 | if (!has_edp_encoder || | ||
| 4805 | intel_encoder_is_pch_edp(&has_edp_encoder->base)) | ||
| 4806 | refclk = 120000; /* 120Mhz refclk */ | ||
| 4807 | } | ||
| 4808 | |||
| 4809 | /* | ||
| 4810 | * Returns a set of divisors for the desired target clock with the given | ||
| 4811 | * refclk, or FALSE. The returned values represent the clock equation: | ||
| 4812 | * reflck * (5 * (m1 + 2) + (m2 + 2)) / (n + 2) / p1 / p2. | ||
| 4813 | */ | ||
| 4814 | limit = intel_limit(crtc, refclk); | ||
| 4815 | ok = limit->find_pll(limit, crtc, adjusted_mode->clock, refclk, &clock); | ||
| 4816 | if (!ok) { | ||
| 4817 | DRM_ERROR("Couldn't find PLL settings for mode!\n"); | ||
| 4818 | return -EINVAL; | ||
| 4819 | } | ||
| 4820 | |||
| 4821 | /* Ensure that the cursor is valid for the new mode before changing... */ | ||
| 4822 | intel_crtc_update_cursor(crtc, true); | ||
| 4823 | |||
| 4824 | if (is_lvds && dev_priv->lvds_downclock_avail) { | ||
| 4825 | has_reduced_clock = limit->find_pll(limit, crtc, | ||
| 4826 | dev_priv->lvds_downclock, | ||
| 4827 | refclk, | ||
| 4828 | &reduced_clock); | ||
| 4829 | if (has_reduced_clock && (clock.p != reduced_clock.p)) { | ||
| 4830 | /* | ||
| 4831 | * If the different P is found, it means that we can't | ||
| 4832 | * switch the display clock by using the FP0/FP1. | ||
| 4833 | * In such case we will disable the LVDS downclock | ||
| 4834 | * feature. | ||
| 4835 | */ | ||
| 4836 | DRM_DEBUG_KMS("Different P is found for " | ||
| 4837 | "LVDS clock/downclock\n"); | ||
| 4838 | has_reduced_clock = 0; | ||
| 4839 | } | ||
| 4840 | } | ||
| 4841 | /* SDVO TV has fixed PLL values depend on its clock range, | ||
| 4842 | this mirrors vbios setting. */ | ||
| 4843 | if (is_sdvo && is_tv) { | ||
| 4844 | if (adjusted_mode->clock >= 100000 | ||
| 4845 | && adjusted_mode->clock < 140500) { | ||
| 4846 | clock.p1 = 2; | ||
| 4847 | clock.p2 = 10; | ||
| 4848 | clock.n = 3; | ||
| 4849 | clock.m1 = 16; | ||
| 4850 | clock.m2 = 8; | ||
| 4851 | } else if (adjusted_mode->clock >= 140500 | ||
| 4852 | && adjusted_mode->clock <= 200000) { | ||
| 4853 | clock.p1 = 1; | ||
| 4854 | clock.p2 = 10; | ||
| 4855 | clock.n = 6; | ||
| 4856 | clock.m1 = 12; | ||
| 4857 | clock.m2 = 8; | ||
| 4858 | } | ||
| 4859 | } | ||
| 4860 | |||
| 4861 | /* FDI link */ | ||
| 4862 | pixel_multiplier = intel_mode_get_pixel_multiplier(adjusted_mode); | ||
| 4863 | lane = 0; | ||
| 4864 | /* CPU eDP doesn't require FDI link, so just set DP M/N | ||
| 4865 | according to current link config */ | ||
| 4866 | if (has_edp_encoder && | ||
| 4867 | !intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | ||
| 4868 | target_clock = mode->clock; | ||
| 4869 | intel_edp_link_config(has_edp_encoder, | ||
| 4870 | &lane, &link_bw); | ||
| 4871 | } else { | ||
| 4872 | /* [e]DP over FDI requires target mode clock | ||
| 4873 | instead of link clock */ | ||
| 4874 | if (is_dp || intel_encoder_is_pch_edp(&has_edp_encoder->base)) | ||
| 4875 | target_clock = mode->clock; | ||
| 4876 | else | ||
| 4877 | target_clock = adjusted_mode->clock; | ||
| 4878 | |||
| 4879 | /* FDI is a binary signal running at ~2.7GHz, encoding | ||
| 4880 | * each output octet as 10 bits. The actual frequency | ||
| 4881 | * is stored as a divider into a 100MHz clock, and the | ||
| 4882 | * mode pixel clock is stored in units of 1KHz. | ||
| 4883 | * Hence the bw of each lane in terms of the mode signal | ||
| 4884 | * is: | ||
| 4885 | */ | ||
| 4886 | link_bw = intel_fdi_link_freq(dev) * MHz(100)/KHz(1)/10; | ||
| 4887 | } | ||
| 4888 | |||
| 4889 | /* determine panel color depth */ | ||
| 4890 | temp = I915_READ(PIPECONF(pipe)); | ||
| 4891 | temp &= ~PIPE_BPC_MASK; | ||
| 4892 | if (is_lvds) { | ||
| 4893 | /* the BPC will be 6 if it is 18-bit LVDS panel */ | ||
| 4894 | if ((I915_READ(PCH_LVDS) & LVDS_A3_POWER_MASK) == LVDS_A3_POWER_UP) | ||
| 4895 | temp |= PIPE_8BPC; | ||
| 4896 | else | ||
| 4897 | temp |= PIPE_6BPC; | ||
| 4898 | } else if (has_edp_encoder) { | ||
| 4899 | switch (dev_priv->edp.bpp/3) { | ||
| 4900 | case 8: | ||
| 4901 | temp |= PIPE_8BPC; | ||
| 4902 | break; | ||
| 4903 | case 10: | ||
| 4904 | temp |= PIPE_10BPC; | ||
| 4905 | break; | ||
| 4906 | case 6: | ||
| 4907 | temp |= PIPE_6BPC; | ||
| 4908 | break; | ||
| 4909 | case 12: | ||
| 4910 | temp |= PIPE_12BPC; | ||
| 4911 | break; | ||
| 4912 | } | ||
| 4913 | } else | ||
| 4914 | temp |= PIPE_8BPC; | ||
| 4915 | I915_WRITE(PIPECONF(pipe), temp); | ||
| 4916 | |||
| 4917 | switch (temp & PIPE_BPC_MASK) { | ||
| 4918 | case PIPE_8BPC: | ||
| 4919 | bpp = 24; | ||
| 4920 | break; | ||
| 4921 | case PIPE_10BPC: | ||
| 4922 | bpp = 30; | ||
| 4923 | break; | ||
| 4924 | case PIPE_6BPC: | ||
| 4925 | bpp = 18; | ||
| 4926 | break; | ||
| 4927 | case PIPE_12BPC: | ||
| 4928 | bpp = 36; | ||
| 4929 | break; | ||
| 4930 | default: | ||
| 4931 | DRM_ERROR("unknown pipe bpc value\n"); | ||
| 4932 | bpp = 24; | ||
| 4933 | } | ||
| 4934 | |||
| 4935 | if (!lane) { | ||
| 4936 | /* | ||
| 4937 | * Account for spread spectrum to avoid | ||
| 4938 | * oversubscribing the link. Max center spread | ||
| 4939 | * is 2.5%; use 5% for safety's sake. | ||
| 4940 | */ | ||
| 4941 | u32 bps = target_clock * bpp * 21 / 20; | ||
| 4942 | lane = bps / (link_bw * 8) + 1; | ||
| 4943 | } | ||
| 4944 | |||
| 4945 | intel_crtc->fdi_lanes = lane; | ||
| 4946 | |||
| 4947 | if (pixel_multiplier > 1) | ||
| 4948 | link_bw *= pixel_multiplier; | ||
| 4949 | ironlake_compute_m_n(bpp, lane, target_clock, link_bw, &m_n); | ||
| 4950 | |||
| 4951 | /* Ironlake: try to setup display ref clock before DPLL | ||
| 4952 | * enabling. This is only under driver's control after | ||
| 4953 | * PCH B stepping, previous chipset stepping should be | ||
| 4954 | * ignoring this setting. | ||
| 4955 | */ | ||
| 4956 | temp = I915_READ(PCH_DREF_CONTROL); | ||
| 4957 | /* Always enable nonspread source */ | ||
| 4958 | temp &= ~DREF_NONSPREAD_SOURCE_MASK; | ||
| 4959 | temp |= DREF_NONSPREAD_SOURCE_ENABLE; | ||
| 4960 | temp &= ~DREF_SSC_SOURCE_MASK; | ||
| 4961 | temp |= DREF_SSC_SOURCE_ENABLE; | ||
| 4962 | I915_WRITE(PCH_DREF_CONTROL, temp); | ||
| 4963 | |||
| 4964 | POSTING_READ(PCH_DREF_CONTROL); | ||
| 4965 | udelay(200); | ||
| 4966 | |||
| 4967 | if (has_edp_encoder) { | ||
| 4968 | if (intel_panel_use_ssc(dev_priv)) { | ||
| 4969 | temp |= DREF_SSC1_ENABLE; | ||
| 4970 | I915_WRITE(PCH_DREF_CONTROL, temp); | ||
| 4971 | |||
| 4972 | POSTING_READ(PCH_DREF_CONTROL); | ||
| 4973 | udelay(200); | ||
| 4974 | } | ||
| 4975 | temp &= ~DREF_CPU_SOURCE_OUTPUT_MASK; | ||
| 4976 | |||
| 4977 | /* Enable CPU source on CPU attached eDP */ | ||
| 4978 | if (!intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | ||
| 4979 | if (intel_panel_use_ssc(dev_priv)) | ||
| 4980 | temp |= DREF_CPU_SOURCE_OUTPUT_DOWNSPREAD; | ||
| 4981 | else | ||
| 4982 | temp |= DREF_CPU_SOURCE_OUTPUT_NONSPREAD; | ||
| 4983 | } else { | ||
| 4984 | /* Enable SSC on PCH eDP if needed */ | ||
| 4985 | if (intel_panel_use_ssc(dev_priv)) { | ||
| 4986 | DRM_ERROR("enabling SSC on PCH\n"); | ||
| 4987 | temp |= DREF_SUPERSPREAD_SOURCE_ENABLE; | ||
| 4988 | } | ||
| 4989 | } | ||
| 4990 | I915_WRITE(PCH_DREF_CONTROL, temp); | ||
| 4991 | POSTING_READ(PCH_DREF_CONTROL); | ||
| 4992 | udelay(200); | ||
| 4993 | } | ||
| 4994 | |||
| 4995 | fp = clock.n << 16 | clock.m1 << 8 | clock.m2; | ||
| 4996 | if (has_reduced_clock) | ||
| 4997 | fp2 = reduced_clock.n << 16 | reduced_clock.m1 << 8 | | ||
| 4998 | reduced_clock.m2; | ||
| 4999 | |||
| 5000 | /* Enable autotuning of the PLL clock (if permissible) */ | ||
| 5001 | factor = 21; | ||
| 5002 | if (is_lvds) { | ||
| 5003 | if ((intel_panel_use_ssc(dev_priv) && | ||
| 5004 | dev_priv->lvds_ssc_freq == 100) || | ||
| 5005 | (I915_READ(PCH_LVDS) & LVDS_CLKB_POWER_MASK) == LVDS_CLKB_POWER_UP) | ||
| 5006 | factor = 25; | ||
| 5007 | } else if (is_sdvo && is_tv) | ||
| 5008 | factor = 20; | ||
| 5009 | |||
| 5010 | if (clock.m1 < factor * clock.n) | ||
| 5011 | fp |= FP_CB_TUNE; | ||
| 5012 | |||
| 5013 | dpll = 0; | ||
| 5014 | |||
| 5015 | if (is_lvds) | ||
| 5016 | dpll |= DPLLB_MODE_LVDS; | ||
| 5017 | else | ||
| 5018 | dpll |= DPLLB_MODE_DAC_SERIAL; | ||
| 5019 | if (is_sdvo) { | ||
| 5020 | int pixel_multiplier = intel_mode_get_pixel_multiplier(adjusted_mode); | ||
| 5021 | if (pixel_multiplier > 1) { | ||
| 5022 | dpll |= (pixel_multiplier - 1) << PLL_REF_SDVO_HDMI_MULTIPLIER_SHIFT; | ||
| 5023 | } | ||
| 5024 | dpll |= DPLL_DVO_HIGH_SPEED; | ||
| 5025 | } | ||
| 5026 | if (is_dp || intel_encoder_is_pch_edp(&has_edp_encoder->base)) | ||
| 5027 | dpll |= DPLL_DVO_HIGH_SPEED; | ||
| 5028 | |||
| 5029 | /* compute bitmask from p1 value */ | ||
| 5030 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA01_P1_POST_DIV_SHIFT; | ||
| 5031 | /* also FPA1 */ | ||
| 5032 | dpll |= (1 << (clock.p1 - 1)) << DPLL_FPA1_P1_POST_DIV_SHIFT; | ||
| 5033 | |||
| 5034 | switch (clock.p2) { | ||
| 5035 | case 5: | ||
| 5036 | dpll |= DPLL_DAC_SERIAL_P2_CLOCK_DIV_5; | ||
| 5037 | break; | ||
| 5038 | case 7: | ||
| 5039 | dpll |= DPLLB_LVDS_P2_CLOCK_DIV_7; | ||
| 5040 | break; | ||
| 5041 | case 10: | ||
| 5042 | dpll |= DPLL_DAC_SERIAL_P2_CLOCK_DIV_10; | ||
| 5043 | break; | ||
| 5044 | case 14: | ||
| 5045 | dpll |= DPLLB_LVDS_P2_CLOCK_DIV_14; | ||
| 5046 | break; | ||
| 5047 | } | ||
| 5048 | |||
| 5049 | if (is_sdvo && is_tv) | ||
| 5050 | dpll |= PLL_REF_INPUT_TVCLKINBC; | ||
| 5051 | else if (is_tv) | ||
| 5052 | /* XXX: just matching BIOS for now */ | ||
| 5053 | /* dpll |= PLL_REF_INPUT_TVCLKINBC; */ | ||
| 5054 | dpll |= 3; | ||
| 5055 | else if (is_lvds && intel_panel_use_ssc(dev_priv) && num_connectors < 2) | ||
| 5056 | dpll |= PLLB_REF_INPUT_SPREADSPECTRUMIN; | ||
| 5057 | else | ||
| 5058 | dpll |= PLL_REF_INPUT_DREFCLK; | ||
| 5059 | |||
| 5060 | /* setup pipeconf */ | ||
| 5061 | pipeconf = I915_READ(PIPECONF(pipe)); | ||
| 5062 | |||
| 5063 | /* Set up the display plane register */ | ||
| 5064 | dspcntr = DISPPLANE_GAMMA_ENABLE; | ||
| 5065 | |||
| 5066 | DRM_DEBUG_KMS("Mode for pipe %c:\n", pipe == 0 ? 'A' : 'B'); | ||
| 5067 | drm_mode_debug_printmodeline(mode); | ||
| 5068 | |||
| 4930 | /* PCH eDP needs FDI, but CPU eDP does not */ | 5069 | /* PCH eDP needs FDI, but CPU eDP does not */ |
| 4931 | if (!has_edp_encoder || intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | 5070 | if (!has_edp_encoder || intel_encoder_is_pch_edp(&has_edp_encoder->base)) { |
| 4932 | I915_WRITE(fp_reg, fp); | 5071 | I915_WRITE(PCH_FP0(pipe), fp); |
| 4933 | I915_WRITE(dpll_reg, dpll & ~DPLL_VCO_ENABLE); | 5072 | I915_WRITE(PCH_DPLL(pipe), dpll & ~DPLL_VCO_ENABLE); |
| 4934 | 5073 | ||
| 4935 | POSTING_READ(dpll_reg); | 5074 | POSTING_READ(PCH_DPLL(pipe)); |
| 4936 | udelay(150); | 5075 | udelay(150); |
| 4937 | } | 5076 | } |
| 4938 | 5077 | ||
| @@ -4964,11 +5103,7 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4964 | * things on. | 5103 | * things on. |
| 4965 | */ | 5104 | */ |
| 4966 | if (is_lvds) { | 5105 | if (is_lvds) { |
| 4967 | reg = LVDS; | 5106 | temp = I915_READ(PCH_LVDS); |
| 4968 | if (HAS_PCH_SPLIT(dev)) | ||
| 4969 | reg = PCH_LVDS; | ||
| 4970 | |||
| 4971 | temp = I915_READ(reg); | ||
| 4972 | temp |= LVDS_PORT_EN | LVDS_A0A2_CLKA_POWER_UP; | 5107 | temp |= LVDS_PORT_EN | LVDS_A0A2_CLKA_POWER_UP; |
| 4973 | if (pipe == 1) { | 5108 | if (pipe == 1) { |
| 4974 | if (HAS_PCH_CPT(dev)) | 5109 | if (HAS_PCH_CPT(dev)) |
| @@ -4995,13 +5130,6 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 4995 | * appropriately here, but we need to look more thoroughly into how | 5130 | * appropriately here, but we need to look more thoroughly into how |
| 4996 | * panels behave in the two modes. | 5131 | * panels behave in the two modes. |
| 4997 | */ | 5132 | */ |
| 4998 | /* set the dithering flag on non-PCH LVDS as needed */ | ||
| 4999 | if (INTEL_INFO(dev)->gen >= 4 && !HAS_PCH_SPLIT(dev)) { | ||
| 5000 | if (dev_priv->lvds_dither) | ||
| 5001 | temp |= LVDS_ENABLE_DITHER; | ||
| 5002 | else | ||
| 5003 | temp &= ~LVDS_ENABLE_DITHER; | ||
| 5004 | } | ||
| 5005 | if (adjusted_mode->flags & DRM_MODE_FLAG_NHSYNC) | 5133 | if (adjusted_mode->flags & DRM_MODE_FLAG_NHSYNC) |
| 5006 | lvds_sync |= LVDS_HSYNC_POLARITY; | 5134 | lvds_sync |= LVDS_HSYNC_POLARITY; |
| 5007 | if (adjusted_mode->flags & DRM_MODE_FLAG_NVSYNC) | 5135 | if (adjusted_mode->flags & DRM_MODE_FLAG_NVSYNC) |
| @@ -5018,22 +5146,20 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 5018 | temp &= ~(LVDS_HSYNC_POLARITY | LVDS_VSYNC_POLARITY); | 5146 | temp &= ~(LVDS_HSYNC_POLARITY | LVDS_VSYNC_POLARITY); |
| 5019 | temp |= lvds_sync; | 5147 | temp |= lvds_sync; |
| 5020 | } | 5148 | } |
| 5021 | I915_WRITE(reg, temp); | 5149 | I915_WRITE(PCH_LVDS, temp); |
| 5022 | } | 5150 | } |
| 5023 | 5151 | ||
| 5024 | /* set the dithering flag and clear for anything other than a panel. */ | 5152 | /* set the dithering flag and clear for anything other than a panel. */ |
| 5025 | if (HAS_PCH_SPLIT(dev)) { | 5153 | pipeconf &= ~PIPECONF_DITHER_EN; |
| 5026 | pipeconf &= ~PIPECONF_DITHER_EN; | 5154 | pipeconf &= ~PIPECONF_DITHER_TYPE_MASK; |
| 5027 | pipeconf &= ~PIPECONF_DITHER_TYPE_MASK; | 5155 | if (dev_priv->lvds_dither && (is_lvds || has_edp_encoder)) { |
| 5028 | if (dev_priv->lvds_dither && (is_lvds || has_edp_encoder)) { | 5156 | pipeconf |= PIPECONF_DITHER_EN; |
| 5029 | pipeconf |= PIPECONF_DITHER_EN; | 5157 | pipeconf |= PIPECONF_DITHER_TYPE_ST1; |
| 5030 | pipeconf |= PIPECONF_DITHER_TYPE_ST1; | ||
| 5031 | } | ||
| 5032 | } | 5158 | } |
| 5033 | 5159 | ||
| 5034 | if (is_dp || intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | 5160 | if (is_dp || intel_encoder_is_pch_edp(&has_edp_encoder->base)) { |
| 5035 | intel_dp_set_m_n(crtc, mode, adjusted_mode); | 5161 | intel_dp_set_m_n(crtc, mode, adjusted_mode); |
| 5036 | } else if (HAS_PCH_SPLIT(dev)) { | 5162 | } else { |
| 5037 | /* For non-DP output, clear any trans DP clock recovery setting.*/ | 5163 | /* For non-DP output, clear any trans DP clock recovery setting.*/ |
| 5038 | I915_WRITE(TRANSDATA_M1(pipe), 0); | 5164 | I915_WRITE(TRANSDATA_M1(pipe), 0); |
| 5039 | I915_WRITE(TRANSDATA_N1(pipe), 0); | 5165 | I915_WRITE(TRANSDATA_N1(pipe), 0); |
| @@ -5041,43 +5167,32 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 5041 | I915_WRITE(TRANSDPLINK_N1(pipe), 0); | 5167 | I915_WRITE(TRANSDPLINK_N1(pipe), 0); |
| 5042 | } | 5168 | } |
| 5043 | 5169 | ||
| 5044 | if (!has_edp_encoder || intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | 5170 | if (!has_edp_encoder || |
| 5045 | I915_WRITE(dpll_reg, dpll); | 5171 | intel_encoder_is_pch_edp(&has_edp_encoder->base)) { |
| 5172 | I915_WRITE(PCH_DPLL(pipe), dpll); | ||
| 5046 | 5173 | ||
| 5047 | /* Wait for the clocks to stabilize. */ | 5174 | /* Wait for the clocks to stabilize. */ |
| 5048 | POSTING_READ(dpll_reg); | 5175 | POSTING_READ(PCH_DPLL(pipe)); |
| 5049 | udelay(150); | 5176 | udelay(150); |
| 5050 | 5177 | ||
| 5051 | if (INTEL_INFO(dev)->gen >= 4 && !HAS_PCH_SPLIT(dev)) { | 5178 | /* The pixel multiplier can only be updated once the |
| 5052 | temp = 0; | 5179 | * DPLL is enabled and the clocks are stable. |
| 5053 | if (is_sdvo) { | 5180 | * |
| 5054 | temp = intel_mode_get_pixel_multiplier(adjusted_mode); | 5181 | * So write it again. |
| 5055 | if (temp > 1) | 5182 | */ |
| 5056 | temp = (temp - 1) << DPLL_MD_UDI_MULTIPLIER_SHIFT; | 5183 | I915_WRITE(PCH_DPLL(pipe), dpll); |
| 5057 | else | ||
| 5058 | temp = 0; | ||
| 5059 | } | ||
| 5060 | I915_WRITE(DPLL_MD(pipe), temp); | ||
| 5061 | } else { | ||
| 5062 | /* The pixel multiplier can only be updated once the | ||
| 5063 | * DPLL is enabled and the clocks are stable. | ||
| 5064 | * | ||
| 5065 | * So write it again. | ||
| 5066 | */ | ||
| 5067 | I915_WRITE(dpll_reg, dpll); | ||
| 5068 | } | ||
| 5069 | } | 5184 | } |
| 5070 | 5185 | ||
| 5071 | intel_crtc->lowfreq_avail = false; | 5186 | intel_crtc->lowfreq_avail = false; |
| 5072 | if (is_lvds && has_reduced_clock && i915_powersave) { | 5187 | if (is_lvds && has_reduced_clock && i915_powersave) { |
| 5073 | I915_WRITE(fp_reg + 4, fp2); | 5188 | I915_WRITE(PCH_FP1(pipe), fp2); |
| 5074 | intel_crtc->lowfreq_avail = true; | 5189 | intel_crtc->lowfreq_avail = true; |
| 5075 | if (HAS_PIPE_CXSR(dev)) { | 5190 | if (HAS_PIPE_CXSR(dev)) { |
| 5076 | DRM_DEBUG_KMS("enabling CxSR downclocking\n"); | 5191 | DRM_DEBUG_KMS("enabling CxSR downclocking\n"); |
| 5077 | pipeconf |= PIPECONF_CXSR_DOWNCLOCK; | 5192 | pipeconf |= PIPECONF_CXSR_DOWNCLOCK; |
| 5078 | } | 5193 | } |
| 5079 | } else { | 5194 | } else { |
| 5080 | I915_WRITE(fp_reg + 4, fp); | 5195 | I915_WRITE(PCH_FP1(pipe), fp); |
| 5081 | if (HAS_PIPE_CXSR(dev)) { | 5196 | if (HAS_PIPE_CXSR(dev)) { |
| 5082 | DRM_DEBUG_KMS("disabling CxSR downclocking\n"); | 5197 | DRM_DEBUG_KMS("disabling CxSR downclocking\n"); |
| 5083 | pipeconf &= ~PIPECONF_CXSR_DOWNCLOCK; | 5198 | pipeconf &= ~PIPECONF_CXSR_DOWNCLOCK; |
| @@ -5116,33 +5231,24 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 5116 | (adjusted_mode->crtc_vsync_start - 1) | | 5231 | (adjusted_mode->crtc_vsync_start - 1) | |
| 5117 | ((adjusted_mode->crtc_vsync_end - 1) << 16)); | 5232 | ((adjusted_mode->crtc_vsync_end - 1) << 16)); |
| 5118 | 5233 | ||
| 5119 | /* pipesrc and dspsize control the size that is scaled from, | 5234 | /* pipesrc controls the size that is scaled from, which should |
| 5120 | * which should always be the user's requested size. | 5235 | * always be the user's requested size. |
| 5121 | */ | 5236 | */ |
| 5122 | if (!HAS_PCH_SPLIT(dev)) { | ||
| 5123 | I915_WRITE(DSPSIZE(plane), | ||
| 5124 | ((mode->vdisplay - 1) << 16) | | ||
| 5125 | (mode->hdisplay - 1)); | ||
| 5126 | I915_WRITE(DSPPOS(plane), 0); | ||
| 5127 | } | ||
| 5128 | I915_WRITE(PIPESRC(pipe), | 5237 | I915_WRITE(PIPESRC(pipe), |
| 5129 | ((mode->hdisplay - 1) << 16) | (mode->vdisplay - 1)); | 5238 | ((mode->hdisplay - 1) << 16) | (mode->vdisplay - 1)); |
| 5130 | 5239 | ||
| 5131 | if (HAS_PCH_SPLIT(dev)) { | 5240 | I915_WRITE(PIPE_DATA_M1(pipe), TU_SIZE(m_n.tu) | m_n.gmch_m); |
| 5132 | I915_WRITE(PIPE_DATA_M1(pipe), TU_SIZE(m_n.tu) | m_n.gmch_m); | 5241 | I915_WRITE(PIPE_DATA_N1(pipe), m_n.gmch_n); |
| 5133 | I915_WRITE(PIPE_DATA_N1(pipe), m_n.gmch_n); | 5242 | I915_WRITE(PIPE_LINK_M1(pipe), m_n.link_m); |
| 5134 | I915_WRITE(PIPE_LINK_M1(pipe), m_n.link_m); | 5243 | I915_WRITE(PIPE_LINK_N1(pipe), m_n.link_n); |
| 5135 | I915_WRITE(PIPE_LINK_N1(pipe), m_n.link_n); | ||
| 5136 | 5244 | ||
| 5137 | if (has_edp_encoder && !intel_encoder_is_pch_edp(&has_edp_encoder->base)) { | 5245 | if (has_edp_encoder && |
| 5138 | ironlake_set_pll_edp(crtc, adjusted_mode->clock); | 5246 | !intel_encoder_is_pch_edp(&has_edp_encoder->base)) { |
| 5139 | } | 5247 | ironlake_set_pll_edp(crtc, adjusted_mode->clock); |
| 5140 | } | 5248 | } |
| 5141 | 5249 | ||
| 5142 | I915_WRITE(PIPECONF(pipe), pipeconf); | 5250 | I915_WRITE(PIPECONF(pipe), pipeconf); |
| 5143 | POSTING_READ(PIPECONF(pipe)); | 5251 | POSTING_READ(PIPECONF(pipe)); |
| 5144 | if (!HAS_PCH_SPLIT(dev)) | ||
| 5145 | intel_enable_pipe(dev_priv, pipe, false); | ||
| 5146 | 5252 | ||
| 5147 | intel_wait_for_vblank(dev, pipe); | 5253 | intel_wait_for_vblank(dev, pipe); |
| 5148 | 5254 | ||
| @@ -5154,13 +5260,31 @@ static int intel_crtc_mode_set(struct drm_crtc *crtc, | |||
| 5154 | 5260 | ||
| 5155 | I915_WRITE(DSPCNTR(plane), dspcntr); | 5261 | I915_WRITE(DSPCNTR(plane), dspcntr); |
| 5156 | POSTING_READ(DSPCNTR(plane)); | 5262 | POSTING_READ(DSPCNTR(plane)); |
| 5157 | if (!HAS_PCH_SPLIT(dev)) | ||
| 5158 | intel_enable_plane(dev_priv, plane, pipe); | ||
| 5159 | 5263 | ||
| 5160 | ret = intel_pipe_set_base(crtc, x, y, old_fb); | 5264 | ret = intel_pipe_set_base(crtc, x, y, old_fb); |
| 5161 | 5265 | ||
| 5162 | intel_update_watermarks(dev); | 5266 | intel_update_watermarks(dev); |
| 5163 | 5267 | ||
| 5268 | return ret; | ||
| 5269 | } | ||
| 5270 | |||
| 5271 | static int intel_crtc_mode_set(struct drm_crtc *crtc, | ||
| 5272 | struct drm_display_mode *mode, | ||
| 5273 | struct drm_display_mode *adjusted_mode, | ||
| 5274 | int x, int y, | ||
| 5275 | struct drm_framebuffer *old_fb) | ||
| 5276 | { | ||
| 5277 | struct drm_device *dev = crtc->dev; | ||
| 5278 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 5279 | struct intel_crtc *intel_crtc = to_intel_crtc(crtc); | ||
| 5280 | int pipe = intel_crtc->pipe; | ||
| 5281 | int ret; | ||
| 5282 | |||
| 5283 | drm_vblank_pre_modeset(dev, pipe); | ||
| 5284 | |||
| 5285 | ret = dev_priv->display.crtc_mode_set(crtc, mode, adjusted_mode, | ||
| 5286 | x, y, old_fb); | ||
| 5287 | |||
| 5164 | drm_vblank_post_modeset(dev, pipe); | 5288 | drm_vblank_post_modeset(dev, pipe); |
| 5165 | 5289 | ||
| 5166 | return ret; | 5290 | return ret; |
| @@ -5483,43 +5607,140 @@ static struct drm_display_mode load_detect_mode = { | |||
| 5483 | 704, 832, 0, 480, 489, 491, 520, 0, DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC), | 5607 | 704, 832, 0, 480, 489, 491, 520, 0, DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC), |
| 5484 | }; | 5608 | }; |
| 5485 | 5609 | ||
| 5486 | struct drm_crtc *intel_get_load_detect_pipe(struct intel_encoder *intel_encoder, | 5610 | static struct drm_framebuffer * |
| 5487 | struct drm_connector *connector, | 5611 | intel_framebuffer_create(struct drm_device *dev, |
| 5488 | struct drm_display_mode *mode, | 5612 | struct drm_mode_fb_cmd *mode_cmd, |
| 5489 | int *dpms_mode) | 5613 | struct drm_i915_gem_object *obj) |
| 5614 | { | ||
| 5615 | struct intel_framebuffer *intel_fb; | ||
| 5616 | int ret; | ||
| 5617 | |||
| 5618 | intel_fb = kzalloc(sizeof(*intel_fb), GFP_KERNEL); | ||
| 5619 | if (!intel_fb) { | ||
| 5620 | drm_gem_object_unreference_unlocked(&obj->base); | ||
| 5621 | return ERR_PTR(-ENOMEM); | ||
| 5622 | } | ||
| 5623 | |||
| 5624 | ret = intel_framebuffer_init(dev, intel_fb, mode_cmd, obj); | ||
| 5625 | if (ret) { | ||
| 5626 | drm_gem_object_unreference_unlocked(&obj->base); | ||
| 5627 | kfree(intel_fb); | ||
| 5628 | return ERR_PTR(ret); | ||
| 5629 | } | ||
| 5630 | |||
| 5631 | return &intel_fb->base; | ||
| 5632 | } | ||
| 5633 | |||
| 5634 | static u32 | ||
| 5635 | intel_framebuffer_pitch_for_width(int width, int bpp) | ||
| 5636 | { | ||
| 5637 | u32 pitch = DIV_ROUND_UP(width * bpp, 8); | ||
| 5638 | return ALIGN(pitch, 64); | ||
| 5639 | } | ||
| 5640 | |||
| 5641 | static u32 | ||
| 5642 | intel_framebuffer_size_for_mode(struct drm_display_mode *mode, int bpp) | ||
| 5643 | { | ||
| 5644 | u32 pitch = intel_framebuffer_pitch_for_width(mode->hdisplay, bpp); | ||
| 5645 | return ALIGN(pitch * mode->vdisplay, PAGE_SIZE); | ||
| 5646 | } | ||
| 5647 | |||
| 5648 | static struct drm_framebuffer * | ||
| 5649 | intel_framebuffer_create_for_mode(struct drm_device *dev, | ||
| 5650 | struct drm_display_mode *mode, | ||
| 5651 | int depth, int bpp) | ||
| 5652 | { | ||
| 5653 | struct drm_i915_gem_object *obj; | ||
| 5654 | struct drm_mode_fb_cmd mode_cmd; | ||
| 5655 | |||
| 5656 | obj = i915_gem_alloc_object(dev, | ||
| 5657 | intel_framebuffer_size_for_mode(mode, bpp)); | ||
| 5658 | if (obj == NULL) | ||
| 5659 | return ERR_PTR(-ENOMEM); | ||
| 5660 | |||
| 5661 | mode_cmd.width = mode->hdisplay; | ||
| 5662 | mode_cmd.height = mode->vdisplay; | ||
| 5663 | mode_cmd.depth = depth; | ||
| 5664 | mode_cmd.bpp = bpp; | ||
| 5665 | mode_cmd.pitch = intel_framebuffer_pitch_for_width(mode_cmd.width, bpp); | ||
| 5666 | |||
| 5667 | return intel_framebuffer_create(dev, &mode_cmd, obj); | ||
| 5668 | } | ||
| 5669 | |||
| 5670 | static struct drm_framebuffer * | ||
| 5671 | mode_fits_in_fbdev(struct drm_device *dev, | ||
| 5672 | struct drm_display_mode *mode) | ||
| 5673 | { | ||
| 5674 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 5675 | struct drm_i915_gem_object *obj; | ||
| 5676 | struct drm_framebuffer *fb; | ||
| 5677 | |||
| 5678 | if (dev_priv->fbdev == NULL) | ||
| 5679 | return NULL; | ||
| 5680 | |||
| 5681 | obj = dev_priv->fbdev->ifb.obj; | ||
| 5682 | if (obj == NULL) | ||
| 5683 | return NULL; | ||
| 5684 | |||
| 5685 | fb = &dev_priv->fbdev->ifb.base; | ||
| 5686 | if (fb->pitch < intel_framebuffer_pitch_for_width(mode->hdisplay, | ||
| 5687 | fb->bits_per_pixel)) | ||
| 5688 | return NULL; | ||
| 5689 | |||
| 5690 | if (obj->base.size < mode->vdisplay * fb->pitch) | ||
| 5691 | return NULL; | ||
| 5692 | |||
| 5693 | return fb; | ||
| 5694 | } | ||
| 5695 | |||
| 5696 | bool intel_get_load_detect_pipe(struct intel_encoder *intel_encoder, | ||
| 5697 | struct drm_connector *connector, | ||
| 5698 | struct drm_display_mode *mode, | ||
| 5699 | struct intel_load_detect_pipe *old) | ||
| 5490 | { | 5700 | { |
| 5491 | struct intel_crtc *intel_crtc; | 5701 | struct intel_crtc *intel_crtc; |
| 5492 | struct drm_crtc *possible_crtc; | 5702 | struct drm_crtc *possible_crtc; |
| 5493 | struct drm_crtc *supported_crtc =NULL; | ||
| 5494 | struct drm_encoder *encoder = &intel_encoder->base; | 5703 | struct drm_encoder *encoder = &intel_encoder->base; |
| 5495 | struct drm_crtc *crtc = NULL; | 5704 | struct drm_crtc *crtc = NULL; |
| 5496 | struct drm_device *dev = encoder->dev; | 5705 | struct drm_device *dev = encoder->dev; |
| 5497 | struct drm_encoder_helper_funcs *encoder_funcs = encoder->helper_private; | 5706 | struct drm_framebuffer *old_fb; |
| 5498 | struct drm_crtc_helper_funcs *crtc_funcs; | ||
| 5499 | int i = -1; | 5707 | int i = -1; |
| 5500 | 5708 | ||
| 5709 | DRM_DEBUG_KMS("[CONNECTOR:%d:%s], [ENCODER:%d:%s]\n", | ||
| 5710 | connector->base.id, drm_get_connector_name(connector), | ||
| 5711 | encoder->base.id, drm_get_encoder_name(encoder)); | ||
| 5712 | |||
| 5501 | /* | 5713 | /* |
| 5502 | * Algorithm gets a little messy: | 5714 | * Algorithm gets a little messy: |
| 5715 | * | ||
| 5503 | * - if the connector already has an assigned crtc, use it (but make | 5716 | * - if the connector already has an assigned crtc, use it (but make |
| 5504 | * sure it's on first) | 5717 | * sure it's on first) |
| 5718 | * | ||
| 5505 | * - try to find the first unused crtc that can drive this connector, | 5719 | * - try to find the first unused crtc that can drive this connector, |
| 5506 | * and use that if we find one | 5720 | * and use that if we find one |
| 5507 | * - if there are no unused crtcs available, try to use the first | ||
| 5508 | * one we found that supports the connector | ||
| 5509 | */ | 5721 | */ |
| 5510 | 5722 | ||
| 5511 | /* See if we already have a CRTC for this connector */ | 5723 | /* See if we already have a CRTC for this connector */ |
| 5512 | if (encoder->crtc) { | 5724 | if (encoder->crtc) { |
| 5513 | crtc = encoder->crtc; | 5725 | crtc = encoder->crtc; |
| 5514 | /* Make sure the crtc and connector are running */ | 5726 | |
| 5515 | intel_crtc = to_intel_crtc(crtc); | 5727 | intel_crtc = to_intel_crtc(crtc); |
| 5516 | *dpms_mode = intel_crtc->dpms_mode; | 5728 | old->dpms_mode = intel_crtc->dpms_mode; |
| 5729 | old->load_detect_temp = false; | ||
| 5730 | |||
| 5731 | /* Make sure the crtc and connector are running */ | ||
| 5517 | if (intel_crtc->dpms_mode != DRM_MODE_DPMS_ON) { | 5732 | if (intel_crtc->dpms_mode != DRM_MODE_DPMS_ON) { |
| 5733 | struct drm_encoder_helper_funcs *encoder_funcs; | ||
| 5734 | struct drm_crtc_helper_funcs *crtc_funcs; | ||
| 5735 | |||
| 5518 | crtc_funcs = crtc->helper_private; | 5736 | crtc_funcs = crtc->helper_private; |
| 5519 | crtc_funcs->dpms(crtc, DRM_MODE_DPMS_ON); | 5737 | crtc_funcs->dpms(crtc, DRM_MODE_DPMS_ON); |
| 5738 | |||
| 5739 | encoder_funcs = encoder->helper_private; | ||
| 5520 | encoder_funcs->dpms(encoder, DRM_MODE_DPMS_ON); | 5740 | encoder_funcs->dpms(encoder, DRM_MODE_DPMS_ON); |
| 5521 | } | 5741 | } |
| 5522 | return crtc; | 5742 | |
| 5743 | return true; | ||
| 5523 | } | 5744 | } |
| 5524 | 5745 | ||
| 5525 | /* Find an unused one (if possible) */ | 5746 | /* Find an unused one (if possible) */ |
| @@ -5531,46 +5752,66 @@ struct drm_crtc *intel_get_load_detect_pipe(struct intel_encoder *intel_encoder, | |||
| 5531 | crtc = possible_crtc; | 5752 | crtc = possible_crtc; |
| 5532 | break; | 5753 | break; |
| 5533 | } | 5754 | } |
| 5534 | if (!supported_crtc) | ||
| 5535 | supported_crtc = possible_crtc; | ||
| 5536 | } | 5755 | } |
| 5537 | 5756 | ||
| 5538 | /* | 5757 | /* |
| 5539 | * If we didn't find an unused CRTC, don't use any. | 5758 | * If we didn't find an unused CRTC, don't use any. |
| 5540 | */ | 5759 | */ |
| 5541 | if (!crtc) { | 5760 | if (!crtc) { |
| 5542 | return NULL; | 5761 | DRM_DEBUG_KMS("no pipe available for load-detect\n"); |
| 5762 | return false; | ||
| 5543 | } | 5763 | } |
| 5544 | 5764 | ||
| 5545 | encoder->crtc = crtc; | 5765 | encoder->crtc = crtc; |
| 5546 | connector->encoder = encoder; | 5766 | connector->encoder = encoder; |
| 5547 | intel_encoder->load_detect_temp = true; | ||
| 5548 | 5767 | ||
| 5549 | intel_crtc = to_intel_crtc(crtc); | 5768 | intel_crtc = to_intel_crtc(crtc); |
| 5550 | *dpms_mode = intel_crtc->dpms_mode; | 5769 | old->dpms_mode = intel_crtc->dpms_mode; |
| 5770 | old->load_detect_temp = true; | ||
| 5771 | old->release_fb = NULL; | ||
| 5551 | 5772 | ||
| 5552 | if (!crtc->enabled) { | 5773 | if (!mode) |
| 5553 | if (!mode) | 5774 | mode = &load_detect_mode; |
| 5554 | mode = &load_detect_mode; | 5775 | |
| 5555 | drm_crtc_helper_set_mode(crtc, mode, 0, 0, crtc->fb); | 5776 | old_fb = crtc->fb; |
| 5556 | } else { | ||
| 5557 | if (intel_crtc->dpms_mode != DRM_MODE_DPMS_ON) { | ||
| 5558 | crtc_funcs = crtc->helper_private; | ||
| 5559 | crtc_funcs->dpms(crtc, DRM_MODE_DPMS_ON); | ||
| 5560 | } | ||
| 5561 | 5777 | ||
| 5562 | /* Add this connector to the crtc */ | 5778 | /* We need a framebuffer large enough to accommodate all accesses |
| 5563 | encoder_funcs->mode_set(encoder, &crtc->mode, &crtc->mode); | 5779 | * that the plane may generate whilst we perform load detection. |
| 5564 | encoder_funcs->commit(encoder); | 5780 | * We can not rely on the fbcon either being present (we get called |
| 5781 | * during its initialisation to detect all boot displays, or it may | ||
| 5782 | * not even exist) or that it is large enough to satisfy the | ||
| 5783 | * requested mode. | ||
| 5784 | */ | ||
| 5785 | crtc->fb = mode_fits_in_fbdev(dev, mode); | ||
| 5786 | if (crtc->fb == NULL) { | ||
| 5787 | DRM_DEBUG_KMS("creating tmp fb for load-detection\n"); | ||
| 5788 | crtc->fb = intel_framebuffer_create_for_mode(dev, mode, 24, 32); | ||
| 5789 | old->release_fb = crtc->fb; | ||
| 5790 | } else | ||
| 5791 | DRM_DEBUG_KMS("reusing fbdev for load-detection framebuffer\n"); | ||
| 5792 | if (IS_ERR(crtc->fb)) { | ||
| 5793 | DRM_DEBUG_KMS("failed to allocate framebuffer for load-detection\n"); | ||
| 5794 | crtc->fb = old_fb; | ||
| 5795 | return false; | ||
| 5565 | } | 5796 | } |
| 5797 | |||
| 5798 | if (!drm_crtc_helper_set_mode(crtc, mode, 0, 0, old_fb)) { | ||
| 5799 | DRM_DEBUG_KMS("failed to set mode on load-detect pipe\n"); | ||
| 5800 | if (old->release_fb) | ||
| 5801 | old->release_fb->funcs->destroy(old->release_fb); | ||
| 5802 | crtc->fb = old_fb; | ||
| 5803 | return false; | ||
| 5804 | } | ||
| 5805 | |||
| 5566 | /* let the connector get through one full cycle before testing */ | 5806 | /* let the connector get through one full cycle before testing */ |
| 5567 | intel_wait_for_vblank(dev, intel_crtc->pipe); | 5807 | intel_wait_for_vblank(dev, intel_crtc->pipe); |
| 5568 | 5808 | ||
| 5569 | return crtc; | 5809 | return true; |
| 5570 | } | 5810 | } |
| 5571 | 5811 | ||
| 5572 | void intel_release_load_detect_pipe(struct intel_encoder *intel_encoder, | 5812 | void intel_release_load_detect_pipe(struct intel_encoder *intel_encoder, |
| 5573 | struct drm_connector *connector, int dpms_mode) | 5813 | struct drm_connector *connector, |
| 5814 | struct intel_load_detect_pipe *old) | ||
| 5574 | { | 5815 | { |
| 5575 | struct drm_encoder *encoder = &intel_encoder->base; | 5816 | struct drm_encoder *encoder = &intel_encoder->base; |
| 5576 | struct drm_device *dev = encoder->dev; | 5817 | struct drm_device *dev = encoder->dev; |
| @@ -5578,19 +5819,24 @@ void intel_release_load_detect_pipe(struct intel_encoder *intel_encoder, | |||
| 5578 | struct drm_encoder_helper_funcs *encoder_funcs = encoder->helper_private; | 5819 | struct drm_encoder_helper_funcs *encoder_funcs = encoder->helper_private; |
| 5579 | struct drm_crtc_helper_funcs *crtc_funcs = crtc->helper_private; | 5820 | struct drm_crtc_helper_funcs *crtc_funcs = crtc->helper_private; |
| 5580 | 5821 | ||
| 5581 | if (intel_encoder->load_detect_temp) { | 5822 | DRM_DEBUG_KMS("[CONNECTOR:%d:%s], [ENCODER:%d:%s]\n", |
| 5582 | encoder->crtc = NULL; | 5823 | connector->base.id, drm_get_connector_name(connector), |
| 5824 | encoder->base.id, drm_get_encoder_name(encoder)); | ||
| 5825 | |||
| 5826 | if (old->load_detect_temp) { | ||
| 5583 | connector->encoder = NULL; | 5827 | connector->encoder = NULL; |
| 5584 | intel_encoder->load_detect_temp = false; | ||
| 5585 | crtc->enabled = drm_helper_crtc_in_use(crtc); | ||
| 5586 | drm_helper_disable_unused_functions(dev); | 5828 | drm_helper_disable_unused_functions(dev); |
| 5829 | |||
| 5830 | if (old->release_fb) | ||
| 5831 | old->release_fb->funcs->destroy(old->release_fb); | ||
| 5832 | |||
| 5833 | return; | ||
| 5587 | } | 5834 | } |
| 5588 | 5835 | ||
| 5589 | /* Switch crtc and encoder back off if necessary */ | 5836 | /* Switch crtc and encoder back off if necessary */ |
| 5590 | if (crtc->enabled && dpms_mode != DRM_MODE_DPMS_ON) { | 5837 | if (old->dpms_mode != DRM_MODE_DPMS_ON) { |
| 5591 | if (encoder->crtc == crtc) | 5838 | encoder_funcs->dpms(encoder, old->dpms_mode); |
| 5592 | encoder_funcs->dpms(encoder, dpms_mode); | 5839 | crtc_funcs->dpms(crtc, old->dpms_mode); |
| 5593 | crtc_funcs->dpms(crtc, dpms_mode); | ||
| 5594 | } | 5840 | } |
| 5595 | } | 5841 | } |
| 5596 | 5842 | ||
| @@ -5605,9 +5851,9 @@ static int intel_crtc_clock_get(struct drm_device *dev, struct drm_crtc *crtc) | |||
| 5605 | intel_clock_t clock; | 5851 | intel_clock_t clock; |
| 5606 | 5852 | ||
| 5607 | if ((dpll & DISPLAY_RATE_SELECT_FPA1) == 0) | 5853 | if ((dpll & DISPLAY_RATE_SELECT_FPA1) == 0) |
| 5608 | fp = FP0(pipe); | 5854 | fp = I915_READ(FP0(pipe)); |
| 5609 | else | 5855 | else |
| 5610 | fp = FP1(pipe); | 5856 | fp = I915_READ(FP1(pipe)); |
| 5611 | 5857 | ||
| 5612 | clock.m1 = (fp & FP_M1_DIV_MASK) >> FP_M1_DIV_SHIFT; | 5858 | clock.m1 = (fp & FP_M1_DIV_MASK) >> FP_M1_DIV_SHIFT; |
| 5613 | if (IS_PINEVIEW(dev)) { | 5859 | if (IS_PINEVIEW(dev)) { |
| @@ -6185,6 +6431,7 @@ static int intel_crtc_page_flip(struct drm_crtc *crtc, | |||
| 6185 | break; | 6431 | break; |
| 6186 | 6432 | ||
| 6187 | case 6: | 6433 | case 6: |
| 6434 | case 7: | ||
| 6188 | OUT_RING(MI_DISPLAY_FLIP | | 6435 | OUT_RING(MI_DISPLAY_FLIP | |
| 6189 | MI_DISPLAY_FLIP_PLANE(intel_crtc->plane)); | 6436 | MI_DISPLAY_FLIP_PLANE(intel_crtc->plane)); |
| 6190 | OUT_RING(fb->pitch | obj->tiling_mode); | 6437 | OUT_RING(fb->pitch | obj->tiling_mode); |
| @@ -6504,6 +6751,9 @@ static void intel_setup_outputs(struct drm_device *dev) | |||
| 6504 | } | 6751 | } |
| 6505 | 6752 | ||
| 6506 | intel_panel_setup_backlight(dev); | 6753 | intel_panel_setup_backlight(dev); |
| 6754 | |||
| 6755 | /* disable all the possible outputs/crtcs before entering KMS mode */ | ||
| 6756 | drm_helper_disable_unused_functions(dev); | ||
| 6507 | } | 6757 | } |
| 6508 | 6758 | ||
| 6509 | static void intel_user_framebuffer_destroy(struct drm_framebuffer *fb) | 6759 | static void intel_user_framebuffer_destroy(struct drm_framebuffer *fb) |
| @@ -6571,25 +6821,12 @@ intel_user_framebuffer_create(struct drm_device *dev, | |||
| 6571 | struct drm_mode_fb_cmd *mode_cmd) | 6821 | struct drm_mode_fb_cmd *mode_cmd) |
| 6572 | { | 6822 | { |
| 6573 | struct drm_i915_gem_object *obj; | 6823 | struct drm_i915_gem_object *obj; |
| 6574 | struct intel_framebuffer *intel_fb; | ||
| 6575 | int ret; | ||
| 6576 | 6824 | ||
| 6577 | obj = to_intel_bo(drm_gem_object_lookup(dev, filp, mode_cmd->handle)); | 6825 | obj = to_intel_bo(drm_gem_object_lookup(dev, filp, mode_cmd->handle)); |
| 6578 | if (&obj->base == NULL) | 6826 | if (&obj->base == NULL) |
| 6579 | return ERR_PTR(-ENOENT); | 6827 | return ERR_PTR(-ENOENT); |
| 6580 | 6828 | ||
| 6581 | intel_fb = kzalloc(sizeof(*intel_fb), GFP_KERNEL); | 6829 | return intel_framebuffer_create(dev, mode_cmd, obj); |
| 6582 | if (!intel_fb) | ||
| 6583 | return ERR_PTR(-ENOMEM); | ||
| 6584 | |||
| 6585 | ret = intel_framebuffer_init(dev, intel_fb, mode_cmd, obj); | ||
| 6586 | if (ret) { | ||
| 6587 | drm_gem_object_unreference_unlocked(&obj->base); | ||
| 6588 | kfree(intel_fb); | ||
| 6589 | return ERR_PTR(ret); | ||
| 6590 | } | ||
| 6591 | |||
| 6592 | return &intel_fb->base; | ||
| 6593 | } | 6830 | } |
| 6594 | 6831 | ||
| 6595 | static const struct drm_mode_config_funcs intel_mode_funcs = { | 6832 | static const struct drm_mode_config_funcs intel_mode_funcs = { |
| @@ -6603,13 +6840,14 @@ intel_alloc_context_page(struct drm_device *dev) | |||
| 6603 | struct drm_i915_gem_object *ctx; | 6840 | struct drm_i915_gem_object *ctx; |
| 6604 | int ret; | 6841 | int ret; |
| 6605 | 6842 | ||
| 6843 | WARN_ON(!mutex_is_locked(&dev->struct_mutex)); | ||
| 6844 | |||
| 6606 | ctx = i915_gem_alloc_object(dev, 4096); | 6845 | ctx = i915_gem_alloc_object(dev, 4096); |
| 6607 | if (!ctx) { | 6846 | if (!ctx) { |
| 6608 | DRM_DEBUG("failed to alloc power context, RC6 disabled\n"); | 6847 | DRM_DEBUG("failed to alloc power context, RC6 disabled\n"); |
| 6609 | return NULL; | 6848 | return NULL; |
| 6610 | } | 6849 | } |
| 6611 | 6850 | ||
| 6612 | mutex_lock(&dev->struct_mutex); | ||
| 6613 | ret = i915_gem_object_pin(ctx, 4096, true); | 6851 | ret = i915_gem_object_pin(ctx, 4096, true); |
| 6614 | if (ret) { | 6852 | if (ret) { |
| 6615 | DRM_ERROR("failed to pin power context: %d\n", ret); | 6853 | DRM_ERROR("failed to pin power context: %d\n", ret); |
| @@ -6621,7 +6859,6 @@ intel_alloc_context_page(struct drm_device *dev) | |||
| 6621 | DRM_ERROR("failed to set-domain on power context: %d\n", ret); | 6859 | DRM_ERROR("failed to set-domain on power context: %d\n", ret); |
| 6622 | goto err_unpin; | 6860 | goto err_unpin; |
| 6623 | } | 6861 | } |
| 6624 | mutex_unlock(&dev->struct_mutex); | ||
| 6625 | 6862 | ||
| 6626 | return ctx; | 6863 | return ctx; |
| 6627 | 6864 | ||
| @@ -6756,6 +6993,11 @@ void gen6_disable_rps(struct drm_device *dev) | |||
| 6756 | I915_WRITE(GEN6_RPNSWREQ, 1 << 31); | 6993 | I915_WRITE(GEN6_RPNSWREQ, 1 << 31); |
| 6757 | I915_WRITE(GEN6_PMINTRMSK, 0xffffffff); | 6994 | I915_WRITE(GEN6_PMINTRMSK, 0xffffffff); |
| 6758 | I915_WRITE(GEN6_PMIER, 0); | 6995 | I915_WRITE(GEN6_PMIER, 0); |
| 6996 | |||
| 6997 | spin_lock_irq(&dev_priv->rps_lock); | ||
| 6998 | dev_priv->pm_iir = 0; | ||
| 6999 | spin_unlock_irq(&dev_priv->rps_lock); | ||
| 7000 | |||
| 6759 | I915_WRITE(GEN6_PMIIR, I915_READ(GEN6_PMIIR)); | 7001 | I915_WRITE(GEN6_PMIIR, I915_READ(GEN6_PMIIR)); |
| 6760 | } | 7002 | } |
| 6761 | 7003 | ||
| @@ -6849,7 +7091,7 @@ void gen6_enable_rps(struct drm_i915_private *dev_priv) | |||
| 6849 | { | 7091 | { |
| 6850 | u32 rp_state_cap = I915_READ(GEN6_RP_STATE_CAP); | 7092 | u32 rp_state_cap = I915_READ(GEN6_RP_STATE_CAP); |
| 6851 | u32 gt_perf_status = I915_READ(GEN6_GT_PERF_STATUS); | 7093 | u32 gt_perf_status = I915_READ(GEN6_GT_PERF_STATUS); |
| 6852 | u32 pcu_mbox; | 7094 | u32 pcu_mbox, rc6_mask = 0; |
| 6853 | int cur_freq, min_freq, max_freq; | 7095 | int cur_freq, min_freq, max_freq; |
| 6854 | int i; | 7096 | int i; |
| 6855 | 7097 | ||
| @@ -6860,7 +7102,8 @@ void gen6_enable_rps(struct drm_i915_private *dev_priv) | |||
| 6860 | * userspace... | 7102 | * userspace... |
| 6861 | */ | 7103 | */ |
| 6862 | I915_WRITE(GEN6_RC_STATE, 0); | 7104 | I915_WRITE(GEN6_RC_STATE, 0); |
| 6863 | __gen6_gt_force_wake_get(dev_priv); | 7105 | mutex_lock(&dev_priv->dev->struct_mutex); |
| 7106 | gen6_gt_force_wake_get(dev_priv); | ||
| 6864 | 7107 | ||
| 6865 | /* disable the counters and set deterministic thresholds */ | 7108 | /* disable the counters and set deterministic thresholds */ |
| 6866 | I915_WRITE(GEN6_RC_CONTROL, 0); | 7109 | I915_WRITE(GEN6_RC_CONTROL, 0); |
| @@ -6880,9 +7123,12 @@ void gen6_enable_rps(struct drm_i915_private *dev_priv) | |||
| 6880 | I915_WRITE(GEN6_RC6p_THRESHOLD, 100000); | 7123 | I915_WRITE(GEN6_RC6p_THRESHOLD, 100000); |
| 6881 | I915_WRITE(GEN6_RC6pp_THRESHOLD, 64000); /* unused */ | 7124 | I915_WRITE(GEN6_RC6pp_THRESHOLD, 64000); /* unused */ |
| 6882 | 7125 | ||
| 7126 | if (i915_enable_rc6) | ||
| 7127 | rc6_mask = GEN6_RC_CTL_RC6p_ENABLE | | ||
| 7128 | GEN6_RC_CTL_RC6_ENABLE; | ||
| 7129 | |||
| 6883 | I915_WRITE(GEN6_RC_CONTROL, | 7130 | I915_WRITE(GEN6_RC_CONTROL, |
| 6884 | GEN6_RC_CTL_RC6p_ENABLE | | 7131 | rc6_mask | |
| 6885 | GEN6_RC_CTL_RC6_ENABLE | | ||
| 6886 | GEN6_RC_CTL_EI_MODE(1) | | 7132 | GEN6_RC_CTL_EI_MODE(1) | |
| 6887 | GEN6_RC_CTL_HW_ENABLE); | 7133 | GEN6_RC_CTL_HW_ENABLE); |
| 6888 | 7134 | ||
| @@ -6954,168 +7200,237 @@ void gen6_enable_rps(struct drm_i915_private *dev_priv) | |||
| 6954 | GEN6_PM_RP_DOWN_THRESHOLD | | 7200 | GEN6_PM_RP_DOWN_THRESHOLD | |
| 6955 | GEN6_PM_RP_UP_EI_EXPIRED | | 7201 | GEN6_PM_RP_UP_EI_EXPIRED | |
| 6956 | GEN6_PM_RP_DOWN_EI_EXPIRED); | 7202 | GEN6_PM_RP_DOWN_EI_EXPIRED); |
| 7203 | spin_lock_irq(&dev_priv->rps_lock); | ||
| 7204 | WARN_ON(dev_priv->pm_iir != 0); | ||
| 6957 | I915_WRITE(GEN6_PMIMR, 0); | 7205 | I915_WRITE(GEN6_PMIMR, 0); |
| 7206 | spin_unlock_irq(&dev_priv->rps_lock); | ||
| 6958 | /* enable all PM interrupts */ | 7207 | /* enable all PM interrupts */ |
| 6959 | I915_WRITE(GEN6_PMINTRMSK, 0); | 7208 | I915_WRITE(GEN6_PMINTRMSK, 0); |
| 6960 | 7209 | ||
| 6961 | __gen6_gt_force_wake_put(dev_priv); | 7210 | gen6_gt_force_wake_put(dev_priv); |
| 7211 | mutex_unlock(&dev_priv->dev->struct_mutex); | ||
| 7212 | } | ||
| 7213 | |||
| 7214 | static void ironlake_init_clock_gating(struct drm_device *dev) | ||
| 7215 | { | ||
| 7216 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7217 | uint32_t dspclk_gate = VRHUNIT_CLOCK_GATE_DISABLE; | ||
| 7218 | |||
| 7219 | /* Required for FBC */ | ||
| 7220 | dspclk_gate |= DPFCUNIT_CLOCK_GATE_DISABLE | | ||
| 7221 | DPFCRUNIT_CLOCK_GATE_DISABLE | | ||
| 7222 | DPFDUNIT_CLOCK_GATE_DISABLE; | ||
| 7223 | /* Required for CxSR */ | ||
| 7224 | dspclk_gate |= DPARBUNIT_CLOCK_GATE_DISABLE; | ||
| 7225 | |||
| 7226 | I915_WRITE(PCH_3DCGDIS0, | ||
| 7227 | MARIUNIT_CLOCK_GATE_DISABLE | | ||
| 7228 | SVSMUNIT_CLOCK_GATE_DISABLE); | ||
| 7229 | I915_WRITE(PCH_3DCGDIS1, | ||
| 7230 | VFMUNIT_CLOCK_GATE_DISABLE); | ||
| 7231 | |||
| 7232 | I915_WRITE(PCH_DSPCLK_GATE_D, dspclk_gate); | ||
| 7233 | |||
| 7234 | /* | ||
| 7235 | * According to the spec the following bits should be set in | ||
| 7236 | * order to enable memory self-refresh | ||
| 7237 | * The bit 22/21 of 0x42004 | ||
| 7238 | * The bit 5 of 0x42020 | ||
| 7239 | * The bit 15 of 0x45000 | ||
| 7240 | */ | ||
| 7241 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7242 | (I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7243 | ILK_DPARB_GATE | ILK_VSDPFD_FULL)); | ||
| 7244 | I915_WRITE(ILK_DSPCLK_GATE, | ||
| 7245 | (I915_READ(ILK_DSPCLK_GATE) | | ||
| 7246 | ILK_DPARB_CLK_GATE)); | ||
| 7247 | I915_WRITE(DISP_ARB_CTL, | ||
| 7248 | (I915_READ(DISP_ARB_CTL) | | ||
| 7249 | DISP_FBC_WM_DIS)); | ||
| 7250 | I915_WRITE(WM3_LP_ILK, 0); | ||
| 7251 | I915_WRITE(WM2_LP_ILK, 0); | ||
| 7252 | I915_WRITE(WM1_LP_ILK, 0); | ||
| 7253 | |||
| 7254 | /* | ||
| 7255 | * Based on the document from hardware guys the following bits | ||
| 7256 | * should be set unconditionally in order to enable FBC. | ||
| 7257 | * The bit 22 of 0x42000 | ||
| 7258 | * The bit 22 of 0x42004 | ||
| 7259 | * The bit 7,8,9 of 0x42020. | ||
| 7260 | */ | ||
| 7261 | if (IS_IRONLAKE_M(dev)) { | ||
| 7262 | I915_WRITE(ILK_DISPLAY_CHICKEN1, | ||
| 7263 | I915_READ(ILK_DISPLAY_CHICKEN1) | | ||
| 7264 | ILK_FBCQ_DIS); | ||
| 7265 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7266 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7267 | ILK_DPARB_GATE); | ||
| 7268 | I915_WRITE(ILK_DSPCLK_GATE, | ||
| 7269 | I915_READ(ILK_DSPCLK_GATE) | | ||
| 7270 | ILK_DPFC_DIS1 | | ||
| 7271 | ILK_DPFC_DIS2 | | ||
| 7272 | ILK_CLK_FBC); | ||
| 7273 | } | ||
| 7274 | |||
| 7275 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7276 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7277 | ILK_ELPIN_409_SELECT); | ||
| 7278 | I915_WRITE(_3D_CHICKEN2, | ||
| 7279 | _3D_CHICKEN2_WM_READ_PIPELINED << 16 | | ||
| 7280 | _3D_CHICKEN2_WM_READ_PIPELINED); | ||
| 6962 | } | 7281 | } |
| 6963 | 7282 | ||
| 6964 | void intel_enable_clock_gating(struct drm_device *dev) | 7283 | static void gen6_init_clock_gating(struct drm_device *dev) |
| 6965 | { | 7284 | { |
| 6966 | struct drm_i915_private *dev_priv = dev->dev_private; | 7285 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 6967 | int pipe; | 7286 | int pipe; |
| 7287 | uint32_t dspclk_gate = VRHUNIT_CLOCK_GATE_DISABLE; | ||
| 7288 | |||
| 7289 | I915_WRITE(PCH_DSPCLK_GATE_D, dspclk_gate); | ||
| 7290 | |||
| 7291 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7292 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7293 | ILK_ELPIN_409_SELECT); | ||
| 7294 | |||
| 7295 | I915_WRITE(WM3_LP_ILK, 0); | ||
| 7296 | I915_WRITE(WM2_LP_ILK, 0); | ||
| 7297 | I915_WRITE(WM1_LP_ILK, 0); | ||
| 6968 | 7298 | ||
| 6969 | /* | 7299 | /* |
| 6970 | * Disable clock gating reported to work incorrectly according to the | 7300 | * According to the spec the following bits should be |
| 6971 | * specs, but enable as much else as we can. | 7301 | * set in order to enable memory self-refresh and fbc: |
| 7302 | * The bit21 and bit22 of 0x42000 | ||
| 7303 | * The bit21 and bit22 of 0x42004 | ||
| 7304 | * The bit5 and bit7 of 0x42020 | ||
| 7305 | * The bit14 of 0x70180 | ||
| 7306 | * The bit14 of 0x71180 | ||
| 6972 | */ | 7307 | */ |
| 6973 | if (HAS_PCH_SPLIT(dev)) { | 7308 | I915_WRITE(ILK_DISPLAY_CHICKEN1, |
| 6974 | uint32_t dspclk_gate = VRHUNIT_CLOCK_GATE_DISABLE; | 7309 | I915_READ(ILK_DISPLAY_CHICKEN1) | |
| 7310 | ILK_FBCQ_DIS | ILK_PABSTRETCH_DIS); | ||
| 7311 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7312 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7313 | ILK_DPARB_GATE | ILK_VSDPFD_FULL); | ||
| 7314 | I915_WRITE(ILK_DSPCLK_GATE, | ||
| 7315 | I915_READ(ILK_DSPCLK_GATE) | | ||
| 7316 | ILK_DPARB_CLK_GATE | | ||
| 7317 | ILK_DPFD_CLK_GATE); | ||
| 6975 | 7318 | ||
| 6976 | if (IS_GEN5(dev)) { | 7319 | for_each_pipe(pipe) |
| 6977 | /* Required for FBC */ | 7320 | I915_WRITE(DSPCNTR(pipe), |
| 6978 | dspclk_gate |= DPFCUNIT_CLOCK_GATE_DISABLE | | 7321 | I915_READ(DSPCNTR(pipe)) | |
| 6979 | DPFCRUNIT_CLOCK_GATE_DISABLE | | 7322 | DISPPLANE_TRICKLE_FEED_DISABLE); |
| 6980 | DPFDUNIT_CLOCK_GATE_DISABLE; | 7323 | } |
| 6981 | /* Required for CxSR */ | ||
| 6982 | dspclk_gate |= DPARBUNIT_CLOCK_GATE_DISABLE; | ||
| 6983 | |||
| 6984 | I915_WRITE(PCH_3DCGDIS0, | ||
| 6985 | MARIUNIT_CLOCK_GATE_DISABLE | | ||
| 6986 | SVSMUNIT_CLOCK_GATE_DISABLE); | ||
| 6987 | I915_WRITE(PCH_3DCGDIS1, | ||
| 6988 | VFMUNIT_CLOCK_GATE_DISABLE); | ||
| 6989 | } | ||
| 6990 | 7324 | ||
| 6991 | I915_WRITE(PCH_DSPCLK_GATE_D, dspclk_gate); | 7325 | static void ivybridge_init_clock_gating(struct drm_device *dev) |
| 7326 | { | ||
| 7327 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7328 | int pipe; | ||
| 7329 | uint32_t dspclk_gate = VRHUNIT_CLOCK_GATE_DISABLE; | ||
| 6992 | 7330 | ||
| 6993 | /* | 7331 | I915_WRITE(PCH_DSPCLK_GATE_D, dspclk_gate); |
| 6994 | * On Ibex Peak and Cougar Point, we need to disable clock | ||
| 6995 | * gating for the panel power sequencer or it will fail to | ||
| 6996 | * start up when no ports are active. | ||
| 6997 | */ | ||
| 6998 | I915_WRITE(SOUTH_DSPCLK_GATE_D, PCH_DPLSUNIT_CLOCK_GATE_DISABLE); | ||
| 6999 | 7332 | ||
| 7000 | /* | 7333 | I915_WRITE(WM3_LP_ILK, 0); |
| 7001 | * According to the spec the following bits should be set in | 7334 | I915_WRITE(WM2_LP_ILK, 0); |
| 7002 | * order to enable memory self-refresh | 7335 | I915_WRITE(WM1_LP_ILK, 0); |
| 7003 | * The bit 22/21 of 0x42004 | ||
| 7004 | * The bit 5 of 0x42020 | ||
| 7005 | * The bit 15 of 0x45000 | ||
| 7006 | */ | ||
| 7007 | if (IS_GEN5(dev)) { | ||
| 7008 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7009 | (I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7010 | ILK_DPARB_GATE | ILK_VSDPFD_FULL)); | ||
| 7011 | I915_WRITE(ILK_DSPCLK_GATE, | ||
| 7012 | (I915_READ(ILK_DSPCLK_GATE) | | ||
| 7013 | ILK_DPARB_CLK_GATE)); | ||
| 7014 | I915_WRITE(DISP_ARB_CTL, | ||
| 7015 | (I915_READ(DISP_ARB_CTL) | | ||
| 7016 | DISP_FBC_WM_DIS)); | ||
| 7017 | I915_WRITE(WM3_LP_ILK, 0); | ||
| 7018 | I915_WRITE(WM2_LP_ILK, 0); | ||
| 7019 | I915_WRITE(WM1_LP_ILK, 0); | ||
| 7020 | } | ||
| 7021 | /* | ||
| 7022 | * Based on the document from hardware guys the following bits | ||
| 7023 | * should be set unconditionally in order to enable FBC. | ||
| 7024 | * The bit 22 of 0x42000 | ||
| 7025 | * The bit 22 of 0x42004 | ||
| 7026 | * The bit 7,8,9 of 0x42020. | ||
| 7027 | */ | ||
| 7028 | if (IS_IRONLAKE_M(dev)) { | ||
| 7029 | I915_WRITE(ILK_DISPLAY_CHICKEN1, | ||
| 7030 | I915_READ(ILK_DISPLAY_CHICKEN1) | | ||
| 7031 | ILK_FBCQ_DIS); | ||
| 7032 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7033 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7034 | ILK_DPARB_GATE); | ||
| 7035 | I915_WRITE(ILK_DSPCLK_GATE, | ||
| 7036 | I915_READ(ILK_DSPCLK_GATE) | | ||
| 7037 | ILK_DPFC_DIS1 | | ||
| 7038 | ILK_DPFC_DIS2 | | ||
| 7039 | ILK_CLK_FBC); | ||
| 7040 | } | ||
| 7041 | 7336 | ||
| 7042 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | 7337 | I915_WRITE(ILK_DSPCLK_GATE, IVB_VRHUNIT_CLK_GATE); |
| 7043 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7044 | ILK_ELPIN_409_SELECT); | ||
| 7045 | 7338 | ||
| 7046 | if (IS_GEN5(dev)) { | 7339 | for_each_pipe(pipe) |
| 7047 | I915_WRITE(_3D_CHICKEN2, | 7340 | I915_WRITE(DSPCNTR(pipe), |
| 7048 | _3D_CHICKEN2_WM_READ_PIPELINED << 16 | | 7341 | I915_READ(DSPCNTR(pipe)) | |
| 7049 | _3D_CHICKEN2_WM_READ_PIPELINED); | 7342 | DISPPLANE_TRICKLE_FEED_DISABLE); |
| 7050 | } | 7343 | } |
| 7051 | 7344 | ||
| 7052 | if (IS_GEN6(dev)) { | 7345 | static void g4x_init_clock_gating(struct drm_device *dev) |
| 7053 | I915_WRITE(WM3_LP_ILK, 0); | 7346 | { |
| 7054 | I915_WRITE(WM2_LP_ILK, 0); | 7347 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 7055 | I915_WRITE(WM1_LP_ILK, 0); | 7348 | uint32_t dspclk_gate; |
| 7056 | 7349 | ||
| 7057 | /* | 7350 | I915_WRITE(RENCLK_GATE_D1, 0); |
| 7058 | * According to the spec the following bits should be | 7351 | I915_WRITE(RENCLK_GATE_D2, VF_UNIT_CLOCK_GATE_DISABLE | |
| 7059 | * set in order to enable memory self-refresh and fbc: | 7352 | GS_UNIT_CLOCK_GATE_DISABLE | |
| 7060 | * The bit21 and bit22 of 0x42000 | 7353 | CL_UNIT_CLOCK_GATE_DISABLE); |
| 7061 | * The bit21 and bit22 of 0x42004 | 7354 | I915_WRITE(RAMCLK_GATE_D, 0); |
| 7062 | * The bit5 and bit7 of 0x42020 | 7355 | dspclk_gate = VRHUNIT_CLOCK_GATE_DISABLE | |
| 7063 | * The bit14 of 0x70180 | 7356 | OVRUNIT_CLOCK_GATE_DISABLE | |
| 7064 | * The bit14 of 0x71180 | 7357 | OVCUNIT_CLOCK_GATE_DISABLE; |
| 7065 | */ | 7358 | if (IS_GM45(dev)) |
| 7066 | I915_WRITE(ILK_DISPLAY_CHICKEN1, | 7359 | dspclk_gate |= DSSUNIT_CLOCK_GATE_DISABLE; |
| 7067 | I915_READ(ILK_DISPLAY_CHICKEN1) | | 7360 | I915_WRITE(DSPCLK_GATE_D, dspclk_gate); |
| 7068 | ILK_FBCQ_DIS | ILK_PABSTRETCH_DIS); | 7361 | } |
| 7069 | I915_WRITE(ILK_DISPLAY_CHICKEN2, | ||
| 7070 | I915_READ(ILK_DISPLAY_CHICKEN2) | | ||
| 7071 | ILK_DPARB_GATE | ILK_VSDPFD_FULL); | ||
| 7072 | I915_WRITE(ILK_DSPCLK_GATE, | ||
| 7073 | I915_READ(ILK_DSPCLK_GATE) | | ||
| 7074 | ILK_DPARB_CLK_GATE | | ||
| 7075 | ILK_DPFD_CLK_GATE); | ||
| 7076 | |||
| 7077 | for_each_pipe(pipe) | ||
| 7078 | I915_WRITE(DSPCNTR(pipe), | ||
| 7079 | I915_READ(DSPCNTR(pipe)) | | ||
| 7080 | DISPPLANE_TRICKLE_FEED_DISABLE); | ||
| 7081 | } | ||
| 7082 | } else if (IS_G4X(dev)) { | ||
| 7083 | uint32_t dspclk_gate; | ||
| 7084 | I915_WRITE(RENCLK_GATE_D1, 0); | ||
| 7085 | I915_WRITE(RENCLK_GATE_D2, VF_UNIT_CLOCK_GATE_DISABLE | | ||
| 7086 | GS_UNIT_CLOCK_GATE_DISABLE | | ||
| 7087 | CL_UNIT_CLOCK_GATE_DISABLE); | ||
| 7088 | I915_WRITE(RAMCLK_GATE_D, 0); | ||
| 7089 | dspclk_gate = VRHUNIT_CLOCK_GATE_DISABLE | | ||
| 7090 | OVRUNIT_CLOCK_GATE_DISABLE | | ||
| 7091 | OVCUNIT_CLOCK_GATE_DISABLE; | ||
| 7092 | if (IS_GM45(dev)) | ||
| 7093 | dspclk_gate |= DSSUNIT_CLOCK_GATE_DISABLE; | ||
| 7094 | I915_WRITE(DSPCLK_GATE_D, dspclk_gate); | ||
| 7095 | } else if (IS_CRESTLINE(dev)) { | ||
| 7096 | I915_WRITE(RENCLK_GATE_D1, I965_RCC_CLOCK_GATE_DISABLE); | ||
| 7097 | I915_WRITE(RENCLK_GATE_D2, 0); | ||
| 7098 | I915_WRITE(DSPCLK_GATE_D, 0); | ||
| 7099 | I915_WRITE(RAMCLK_GATE_D, 0); | ||
| 7100 | I915_WRITE16(DEUC, 0); | ||
| 7101 | } else if (IS_BROADWATER(dev)) { | ||
| 7102 | I915_WRITE(RENCLK_GATE_D1, I965_RCZ_CLOCK_GATE_DISABLE | | ||
| 7103 | I965_RCC_CLOCK_GATE_DISABLE | | ||
| 7104 | I965_RCPB_CLOCK_GATE_DISABLE | | ||
| 7105 | I965_ISC_CLOCK_GATE_DISABLE | | ||
| 7106 | I965_FBC_CLOCK_GATE_DISABLE); | ||
| 7107 | I915_WRITE(RENCLK_GATE_D2, 0); | ||
| 7108 | } else if (IS_GEN3(dev)) { | ||
| 7109 | u32 dstate = I915_READ(D_STATE); | ||
| 7110 | 7362 | ||
| 7111 | dstate |= DSTATE_PLL_D3_OFF | DSTATE_GFX_CLOCK_GATING | | 7363 | static void crestline_init_clock_gating(struct drm_device *dev) |
| 7112 | DSTATE_DOT_CLOCK_GATING; | 7364 | { |
| 7113 | I915_WRITE(D_STATE, dstate); | 7365 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 7114 | } else if (IS_I85X(dev) || IS_I865G(dev)) { | 7366 | |
| 7115 | I915_WRITE(RENCLK_GATE_D1, SV_CLOCK_GATE_DISABLE); | 7367 | I915_WRITE(RENCLK_GATE_D1, I965_RCC_CLOCK_GATE_DISABLE); |
| 7116 | } else if (IS_I830(dev)) { | 7368 | I915_WRITE(RENCLK_GATE_D2, 0); |
| 7117 | I915_WRITE(DSPCLK_GATE_D, OVRUNIT_CLOCK_GATE_DISABLE); | 7369 | I915_WRITE(DSPCLK_GATE_D, 0); |
| 7118 | } | 7370 | I915_WRITE(RAMCLK_GATE_D, 0); |
| 7371 | I915_WRITE16(DEUC, 0); | ||
| 7372 | } | ||
| 7373 | |||
| 7374 | static void broadwater_init_clock_gating(struct drm_device *dev) | ||
| 7375 | { | ||
| 7376 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7377 | |||
| 7378 | I915_WRITE(RENCLK_GATE_D1, I965_RCZ_CLOCK_GATE_DISABLE | | ||
| 7379 | I965_RCC_CLOCK_GATE_DISABLE | | ||
| 7380 | I965_RCPB_CLOCK_GATE_DISABLE | | ||
| 7381 | I965_ISC_CLOCK_GATE_DISABLE | | ||
| 7382 | I965_FBC_CLOCK_GATE_DISABLE); | ||
| 7383 | I915_WRITE(RENCLK_GATE_D2, 0); | ||
| 7384 | } | ||
| 7385 | |||
| 7386 | static void gen3_init_clock_gating(struct drm_device *dev) | ||
| 7387 | { | ||
| 7388 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7389 | u32 dstate = I915_READ(D_STATE); | ||
| 7390 | |||
| 7391 | dstate |= DSTATE_PLL_D3_OFF | DSTATE_GFX_CLOCK_GATING | | ||
| 7392 | DSTATE_DOT_CLOCK_GATING; | ||
| 7393 | I915_WRITE(D_STATE, dstate); | ||
| 7394 | } | ||
| 7395 | |||
| 7396 | static void i85x_init_clock_gating(struct drm_device *dev) | ||
| 7397 | { | ||
| 7398 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7399 | |||
| 7400 | I915_WRITE(RENCLK_GATE_D1, SV_CLOCK_GATE_DISABLE); | ||
| 7401 | } | ||
| 7402 | |||
| 7403 | static void i830_init_clock_gating(struct drm_device *dev) | ||
| 7404 | { | ||
| 7405 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7406 | |||
| 7407 | I915_WRITE(DSPCLK_GATE_D, OVRUNIT_CLOCK_GATE_DISABLE); | ||
| 7408 | } | ||
| 7409 | |||
| 7410 | static void ibx_init_clock_gating(struct drm_device *dev) | ||
| 7411 | { | ||
| 7412 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7413 | |||
| 7414 | /* | ||
| 7415 | * On Ibex Peak and Cougar Point, we need to disable clock | ||
| 7416 | * gating for the panel power sequencer or it will fail to | ||
| 7417 | * start up when no ports are active. | ||
| 7418 | */ | ||
| 7419 | I915_WRITE(SOUTH_DSPCLK_GATE_D, PCH_DPLSUNIT_CLOCK_GATE_DISABLE); | ||
| 7420 | } | ||
| 7421 | |||
| 7422 | static void cpt_init_clock_gating(struct drm_device *dev) | ||
| 7423 | { | ||
| 7424 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7425 | |||
| 7426 | /* | ||
| 7427 | * On Ibex Peak and Cougar Point, we need to disable clock | ||
| 7428 | * gating for the panel power sequencer or it will fail to | ||
| 7429 | * start up when no ports are active. | ||
| 7430 | */ | ||
| 7431 | I915_WRITE(SOUTH_DSPCLK_GATE_D, PCH_DPLSUNIT_CLOCK_GATE_DISABLE); | ||
| 7432 | I915_WRITE(SOUTH_CHICKEN2, I915_READ(SOUTH_CHICKEN2) | | ||
| 7433 | DPLS_EDP_PPS_FIX_DIS); | ||
| 7119 | } | 7434 | } |
| 7120 | 7435 | ||
| 7121 | static void ironlake_teardown_rc6(struct drm_device *dev) | 7436 | static void ironlake_teardown_rc6(struct drm_device *dev) |
| @@ -7185,9 +7500,12 @@ void ironlake_enable_rc6(struct drm_device *dev) | |||
| 7185 | if (!i915_enable_rc6) | 7500 | if (!i915_enable_rc6) |
| 7186 | return; | 7501 | return; |
| 7187 | 7502 | ||
| 7503 | mutex_lock(&dev->struct_mutex); | ||
| 7188 | ret = ironlake_setup_rc6(dev); | 7504 | ret = ironlake_setup_rc6(dev); |
| 7189 | if (ret) | 7505 | if (ret) { |
| 7506 | mutex_unlock(&dev->struct_mutex); | ||
| 7190 | return; | 7507 | return; |
| 7508 | } | ||
| 7191 | 7509 | ||
| 7192 | /* | 7510 | /* |
| 7193 | * GPU can automatically power down the render unit if given a page | 7511 | * GPU can automatically power down the render unit if given a page |
| @@ -7196,6 +7514,7 @@ void ironlake_enable_rc6(struct drm_device *dev) | |||
| 7196 | ret = BEGIN_LP_RING(6); | 7514 | ret = BEGIN_LP_RING(6); |
| 7197 | if (ret) { | 7515 | if (ret) { |
| 7198 | ironlake_teardown_rc6(dev); | 7516 | ironlake_teardown_rc6(dev); |
| 7517 | mutex_unlock(&dev->struct_mutex); | ||
| 7199 | return; | 7518 | return; |
| 7200 | } | 7519 | } |
| 7201 | 7520 | ||
| @@ -7211,10 +7530,33 @@ void ironlake_enable_rc6(struct drm_device *dev) | |||
| 7211 | OUT_RING(MI_FLUSH); | 7530 | OUT_RING(MI_FLUSH); |
| 7212 | ADVANCE_LP_RING(); | 7531 | ADVANCE_LP_RING(); |
| 7213 | 7532 | ||
| 7533 | /* | ||
| 7534 | * Wait for the command parser to advance past MI_SET_CONTEXT. The HW | ||
| 7535 | * does an implicit flush, combined with MI_FLUSH above, it should be | ||
| 7536 | * safe to assume that renderctx is valid | ||
| 7537 | */ | ||
| 7538 | ret = intel_wait_ring_idle(LP_RING(dev_priv)); | ||
| 7539 | if (ret) { | ||
| 7540 | DRM_ERROR("failed to enable ironlake power power savings\n"); | ||
| 7541 | ironlake_teardown_rc6(dev); | ||
| 7542 | mutex_unlock(&dev->struct_mutex); | ||
| 7543 | return; | ||
| 7544 | } | ||
| 7545 | |||
| 7214 | I915_WRITE(PWRCTXA, dev_priv->pwrctx->gtt_offset | PWRCTX_EN); | 7546 | I915_WRITE(PWRCTXA, dev_priv->pwrctx->gtt_offset | PWRCTX_EN); |
| 7215 | I915_WRITE(RSTDBYCTL, I915_READ(RSTDBYCTL) & ~RCX_SW_EXIT); | 7547 | I915_WRITE(RSTDBYCTL, I915_READ(RSTDBYCTL) & ~RCX_SW_EXIT); |
| 7548 | mutex_unlock(&dev->struct_mutex); | ||
| 7216 | } | 7549 | } |
| 7217 | 7550 | ||
| 7551 | void intel_init_clock_gating(struct drm_device *dev) | ||
| 7552 | { | ||
| 7553 | struct drm_i915_private *dev_priv = dev->dev_private; | ||
| 7554 | |||
| 7555 | dev_priv->display.init_clock_gating(dev); | ||
| 7556 | |||
| 7557 | if (dev_priv->display.init_pch_clock_gating) | ||
| 7558 | dev_priv->display.init_pch_clock_gating(dev); | ||
| 7559 | } | ||
| 7218 | 7560 | ||
| 7219 | /* Set up chip specific display functions */ | 7561 | /* Set up chip specific display functions */ |
| 7220 | static void intel_init_display(struct drm_device *dev) | 7562 | static void intel_init_display(struct drm_device *dev) |
| @@ -7222,10 +7564,13 @@ static void intel_init_display(struct drm_device *dev) | |||
| 7222 | struct drm_i915_private *dev_priv = dev->dev_private; | 7564 | struct drm_i915_private *dev_priv = dev->dev_private; |
| 7223 | 7565 | ||
| 7224 | /* We always want a DPMS function */ | 7566 | /* We always want a DPMS function */ |
| 7225 | if (HAS_PCH_SPLIT(dev)) | 7567 | if (HAS_PCH_SPLIT(dev)) { |
| 7226 | dev_priv->display.dpms = ironlake_crtc_dpms; | 7568 | dev_priv->display.dpms = ironlake_crtc_dpms; |
| 7227 | else | 7569 | dev_priv->display.crtc_mode_set = ironlake_crtc_mode_set; |
| 7570 | } else { | ||
| 7228 | dev_priv->display.dpms = i9xx_crtc_dpms; | 7571 | dev_priv->display.dpms = i9xx_crtc_dpms; |
| 7572 | dev_priv->display.crtc_mode_set = i9xx_crtc_mode_set; | ||
| 7573 | } | ||
| 7229 | 7574 | ||
| 7230 | if (I915_HAS_FBC(dev)) { | 7575 | if (I915_HAS_FBC(dev)) { |
| 7231 | if (HAS_PCH_SPLIT(dev)) { | 7576 | if (HAS_PCH_SPLIT(dev)) { |
| @@ -7269,6 +7614,11 @@ static void intel_init_display(struct drm_device *dev) | |||
| 7269 | 7614 | ||
| 7270 | /* For FIFO watermark updates */ | 7615 | /* For FIFO watermark updates */ |
| 7271 | if (HAS_PCH_SPLIT(dev)) { | 7616 | if (HAS_PCH_SPLIT(dev)) { |
| 7617 | if (HAS_PCH_IBX(dev)) | ||
| 7618 | dev_priv->display.init_pch_clock_gating = ibx_init_clock_gating; | ||
| 7619 | else if (HAS_PCH_CPT(dev)) | ||
| 7620 | dev_priv->display.init_pch_clock_gating = cpt_init_clock_gating; | ||
| 7621 | |||
| 7272 | if (IS_GEN5(dev)) { | 7622 | if (IS_GEN5(dev)) { |
| 7273 | if (I915_READ(MLTR_ILK) & ILK_SRLT_MASK) | 7623 | if (I915_READ(MLTR_ILK) & ILK_SRLT_MASK) |
| 7274 | dev_priv->display.update_wm = ironlake_update_wm; | 7624 | dev_priv->display.update_wm = ironlake_update_wm; |
| @@ -7277,6 +7627,8 @@ static void intel_init_display(struct drm_device *dev) | |||
| 7277 | "Disable CxSR\n"); | 7627 | "Disable CxSR\n"); |
| 7278 | dev_priv->display.update_wm = NULL; | 7628 | dev_priv->display.update_wm = NULL; |
| 7279 | } | 7629 | } |
| 7630 | dev_priv->display.fdi_link_train = ironlake_fdi_link_train; | ||
| 7631 | dev_priv->display.init_clock_gating = ironlake_init_clock_gating; | ||
| 7280 | } else if (IS_GEN6(dev)) { | 7632 | } else if (IS_GEN6(dev)) { |
| 7281 | if (SNB_READ_WM0_LATENCY()) { | 7633 | if (SNB_READ_WM0_LATENCY()) { |
| 7282 | dev_priv->display.update_wm = sandybridge_update_wm; | 7634 | dev_priv->display.update_wm = sandybridge_update_wm; |
| @@ -7285,6 +7637,20 @@ static void intel_init_display(struct drm_device *dev) | |||
| 7285 | "Disable CxSR\n"); | 7637 | "Disable CxSR\n"); |
| 7286 | dev_priv->display.update_wm = NULL; | 7638 | dev_priv->display.update_wm = NULL; |
| 7287 | } | 7639 | } |
| 7640 | dev_priv->display.fdi_link_train = gen6_fdi_link_train; | ||
| 7641 | dev_priv->display.init_clock_gating = gen6_init_clock_gating; | ||
| 7642 | } else if (IS_IVYBRIDGE(dev)) { | ||
| 7643 | /* FIXME: detect B0+ stepping and use auto training */ | ||
| 7644 | dev_priv->display.fdi_link_train = ivb_manual_fdi_link_train; | ||
| 7645 | if (SNB_READ_WM0_LATENCY()) { | ||
| 7646 | dev_priv->display.update_wm = sandybridge_update_wm; | ||
| 7647 | } else { | ||
| 7648 | DRM_DEBUG_KMS("Failed to read display plane latency. " | ||
| 7649 | "Disable CxSR\n"); | ||
| 7650 | dev_priv->display.update_wm = NULL; | ||
| 7651 | } | ||
| 7652 | dev_priv->display.init_clock_gating = ivybridge_init_clock_gating; | ||
| 7653 | |||
| 7288 | } else | 7654 | } else |
| 7289 | dev_priv->display.update_wm = NULL; | 7655 | dev_priv->display.update_wm = NULL; |
| 7290 | } else if (IS_PINEVIEW(dev)) { | 7656 | } else if (IS_PINEVIEW(dev)) { |
| @@ -7302,18 +7668,30 @@ static void intel_init_display(struct drm_device *dev) | |||
| 7302 | dev_priv->display.update_wm = NULL; | 7668 | dev_priv->display.update_wm = NULL; |
| 7303 | } else | 7669 | } else |
| 7304 | dev_priv->display.update_wm = pineview_update_wm; | 7670 | dev_priv->display.update_wm = pineview_update_wm; |
| 7305 | } else if (IS_G4X(dev)) | 7671 | } else if (IS_G4X(dev)) { |
| 7306 | dev_priv->display.update_wm = g4x_update_wm; | 7672 | dev_priv->display.update_wm = g4x_update_wm; |
| 7307 | else if (IS_GEN4(dev)) | 7673 | dev_priv->display.init_clock_gating = g4x_init_clock_gating; |
| 7674 | } else if (IS_GEN4(dev)) { | ||
| 7308 | dev_priv->display.update_wm = i965_update_wm; | 7675 | dev_priv->display.update_wm = i965_update_wm; |
| 7309 | else if (IS_GEN3(dev)) { | 7676 | if (IS_CRESTLINE(dev)) |
| 7677 | dev_priv->display.init_clock_gating = crestline_init_clock_gating; | ||
| 7678 | else if (IS_BROADWATER(dev)) | ||
| 7679 | dev_priv->display.init_clock_gating = broadwater_init_clock_gating; | ||
| 7680 | } else if (IS_GEN3(dev)) { | ||
| 7310 | dev_priv->display.update_wm = i9xx_update_wm; | 7681 | dev_priv->display.update_wm = i9xx_update_wm; |
| 7311 | dev_priv->display.get_fifo_size = i9xx_get_fifo_size; | 7682 | dev_priv->display.get_fifo_size = i9xx_get_fifo_size; |
| 7683 | dev_priv->display.init_clock_gating = gen3_init_clock_gating; | ||
| 7684 | } else if (IS_I865G(dev)) { | ||
| 7685 | dev_priv->display.update_wm = i830_update_wm; | ||
| 7686 | dev_priv->display.init_clock_gating = i85x_init_clock_gating; | ||
| 7687 | dev_priv->display.get_fifo_size = i830_get_fifo_size; | ||
| 7312 | } else if (IS_I85X(dev)) { | 7688 | } else if (IS_I85X(dev)) { |
| 7313 | dev_priv->display.update_wm = i9xx_update_wm; | 7689 | dev_priv->display.update_wm = i9xx_update_wm; |
| 7314 | dev_priv->display.get_fifo_size = i85x_get_fifo_size; | 7690 | dev_priv->display.get_fifo_size = i85x_get_fifo_size; |
| 7691 | dev_priv->display.init_clock_gating = i85x_init_clock_gating; | ||
| 7315 | } else { | 7692 | } else { |
| 7316 | dev_priv->display.update_wm = i830_update_wm; | 7693 | dev_priv->display.update_wm = i830_update_wm; |
| 7694 | dev_priv->display.init_clock_gating = i830_init_clock_gating; | ||
| 7317 | if (IS_845G(dev)) | 7695 | if (IS_845G(dev)) |
| 7318 | dev_priv->display.get_fifo_size = i845_get_fifo_size; | 7696 | dev_priv->display.get_fifo_size = i845_get_fifo_size; |
| 7319 | else | 7697 | else |
| @@ -7439,12 +7817,11 @@ void intel_modeset_init(struct drm_device *dev) | |||
| 7439 | intel_crtc_init(dev, i); | 7817 | intel_crtc_init(dev, i); |
| 7440 | } | 7818 | } |
| 7441 | 7819 | ||
| 7442 | intel_setup_outputs(dev); | ||
| 7443 | |||
| 7444 | intel_enable_clock_gating(dev); | ||
| 7445 | |||
| 7446 | /* Just disable it once at startup */ | 7820 | /* Just disable it once at startup */ |
| 7447 | i915_disable_vga(dev); | 7821 | i915_disable_vga(dev); |
| 7822 | intel_setup_outputs(dev); | ||
| 7823 | |||
| 7824 | intel_init_clock_gating(dev); | ||
| 7448 | 7825 | ||
| 7449 | if (IS_IRONLAKE_M(dev)) { | 7826 | if (IS_IRONLAKE_M(dev)) { |
| 7450 | ironlake_enable_drps(dev); | 7827 | ironlake_enable_drps(dev); |
| @@ -7454,12 +7831,15 @@ void intel_modeset_init(struct drm_device *dev) | |||
| 7454 | if (IS_GEN6(dev)) | 7831 | if (IS_GEN6(dev)) |
| 7455 | gen6_enable_rps(dev_priv); | 7832 | gen6_enable_rps(dev_priv); |
| 7456 | 7833 | ||
| 7457 | if (IS_IRONLAKE_M(dev)) | ||
| 7458 | ironlake_enable_rc6(dev); | ||
| 7459 | |||
| 7460 | INIT_WORK(&dev_priv->idle_work, intel_idle_update); | 7834 | INIT_WORK(&dev_priv->idle_work, intel_idle_update); |
| 7461 | setup_timer(&dev_priv->idle_timer, intel_gpu_idle_timer, | 7835 | setup_timer(&dev_priv->idle_timer, intel_gpu_idle_timer, |
| 7462 | (unsigned long)dev); | 7836 | (unsigned long)dev); |
| 7837 | } | ||
| 7838 | |||
| 7839 | void intel_modeset_gem_init(struct drm_device *dev) | ||
| 7840 | { | ||
| 7841 | if (IS_IRONLAKE_M(dev)) | ||
| 7842 | ironlake_enable_rc6(dev); | ||
| 7463 | 7843 | ||
| 7464 | intel_setup_overlay(dev); | 7844 | intel_setup_overlay(dev); |
| 7465 | } | 7845 | } |
diff --git a/drivers/gpu/drm/i915/intel_dp.c b/drivers/gpu/drm/i915/intel_dp.c index cb8578b7e443..a4d80314e7f8 100644 --- a/drivers/gpu/drm/i915/intel_dp.c +++ b/drivers/gpu/drm/i915/intel_dp.c | |||
| @@ -1470,7 +1470,8 @@ intel_dp_link_down(struct intel_dp *intel_dp) | |||
| 1470 | 1470 | ||
| 1471 | if (!HAS_PCH_CPT(dev) && | 1471 | if (!HAS_PCH_CPT(dev) && |
| 1472 | I915_READ(intel_dp->output_reg) & DP_PIPEB_SELECT) { | 1472 | I915_READ(intel_dp->output_reg) & DP_PIPEB_SELECT) { |
| 1473 | struct intel_crtc *intel_crtc = to_intel_crtc(intel_dp->base.base.crtc); | 1473 | struct drm_crtc *crtc = intel_dp->base.base.crtc; |
| 1474 | |||
| 1474 | /* Hardware workaround: leaving our transcoder select | 1475 | /* Hardware workaround: leaving our transcoder select |
| 1475 | * set to transcoder B while it's off will prevent the | 1476 | * set to transcoder B while it's off will prevent the |
| 1476 | * corresponding HDMI output on transcoder A. | 1477 | * corresponding HDMI output on transcoder A. |
| @@ -1485,7 +1486,19 @@ intel_dp_link_down(struct intel_dp *intel_dp) | |||
| 1485 | /* Changes to enable or select take place the vblank | 1486 | /* Changes to enable or select take place the vblank |
| 1486 | * after being written. | 1487 | * after being written. |
| 1487 | */ | 1488 | */ |
| 1488 | intel_wait_for_vblank(dev, intel_crtc->pipe); | 1489 | if (crtc == NULL) { |
| 1490 | /* We can arrive here never having been attached | ||
| 1491 | * to a CRTC, for instance, due to inheriting | ||
| 1492 | * random state from the BIOS. | ||
| 1493 | * | ||
| 1494 | * If the pipe is not running, play safe and | ||
| 1495 | * wait for the clocks to stabilise before | ||
| 1496 | * continuing. | ||
| 1497 | */ | ||
| 1498 | POSTING_READ(intel_dp->output_reg); | ||
| 1499 | msleep(50); | ||
| 1500 | } else | ||
| 1501 | intel_wait_for_vblank(dev, to_intel_crtc(crtc)->pipe); | ||
| 1489 | } | 1502 | } |
| 1490 | 1503 | ||
| 1491 | I915_WRITE(intel_dp->output_reg, DP & ~DP_PORT_EN); | 1504 | I915_WRITE(intel_dp->output_reg, DP & ~DP_PORT_EN); |
diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h index 1d20712d527f..831d7a4a0d18 100644 --- a/drivers/gpu/drm/i915/intel_drv.h +++ b/drivers/gpu/drm/i915/intel_drv.h | |||
| @@ -140,7 +140,6 @@ struct intel_fbdev { | |||
| 140 | struct intel_encoder { | 140 | struct intel_encoder { |
| 141 | struct drm_encoder base; | 141 | struct drm_encoder base; |
| 142 | int type; | 142 | int type; |
| 143 | bool load_detect_temp; | ||
| 144 | bool needs_tv_clock; | 143 | bool needs_tv_clock; |
| 145 | void (*hot_plug)(struct intel_encoder *); | 144 | void (*hot_plug)(struct intel_encoder *); |
| 146 | int crtc_mask; | 145 | int crtc_mask; |
| @@ -291,13 +290,19 @@ int intel_get_pipe_from_crtc_id(struct drm_device *dev, void *data, | |||
| 291 | struct drm_file *file_priv); | 290 | struct drm_file *file_priv); |
| 292 | extern void intel_wait_for_vblank(struct drm_device *dev, int pipe); | 291 | extern void intel_wait_for_vblank(struct drm_device *dev, int pipe); |
| 293 | extern void intel_wait_for_pipe_off(struct drm_device *dev, int pipe); | 292 | extern void intel_wait_for_pipe_off(struct drm_device *dev, int pipe); |
| 294 | extern struct drm_crtc *intel_get_load_detect_pipe(struct intel_encoder *intel_encoder, | 293 | |
| 295 | struct drm_connector *connector, | 294 | struct intel_load_detect_pipe { |
| 296 | struct drm_display_mode *mode, | 295 | struct drm_framebuffer *release_fb; |
| 297 | int *dpms_mode); | 296 | bool load_detect_temp; |
| 297 | int dpms_mode; | ||
| 298 | }; | ||
| 299 | extern bool intel_get_load_detect_pipe(struct intel_encoder *intel_encoder, | ||
| 300 | struct drm_connector *connector, | ||
| 301 | struct drm_display_mode *mode, | ||
| 302 | struct intel_load_detect_pipe *old); | ||
| 298 | extern void intel_release_load_detect_pipe(struct intel_encoder *intel_encoder, | 303 | extern void intel_release_load_detect_pipe(struct intel_encoder *intel_encoder, |
| 299 | struct drm_connector *connector, | 304 | struct drm_connector *connector, |
| 300 | int dpms_mode); | 305 | struct intel_load_detect_pipe *old); |
| 301 | 306 | ||
| 302 | extern struct drm_connector* intel_sdvo_find(struct drm_device *dev, int sdvoB); | 307 | extern struct drm_connector* intel_sdvo_find(struct drm_device *dev, int sdvoB); |
| 303 | extern int intel_sdvo_supports_hotplug(struct drm_connector *connector); | 308 | extern int intel_sdvo_supports_hotplug(struct drm_connector *connector); |
| @@ -339,4 +344,6 @@ extern int intel_overlay_attrs(struct drm_device *dev, void *data, | |||
| 339 | 344 | ||
| 340 | extern void intel_fb_output_poll_changed(struct drm_device *dev); | 345 | extern void intel_fb_output_poll_changed(struct drm_device *dev); |
| 341 | extern void intel_fb_restore_mode(struct drm_device *dev); | 346 | extern void intel_fb_restore_mode(struct drm_device *dev); |
| 347 | |||
| 348 | extern void intel_init_clock_gating(struct drm_device *dev); | ||
| 342 | #endif /* __INTEL_DRV_H__ */ | 349 | #endif /* __INTEL_DRV_H__ */ |
diff --git a/drivers/gpu/drm/i915/intel_lvds.c b/drivers/gpu/drm/i915/intel_lvds.c index a562bd2648c7..67cb076d271b 100644 --- a/drivers/gpu/drm/i915/intel_lvds.c +++ b/drivers/gpu/drm/i915/intel_lvds.c | |||
| @@ -539,6 +539,9 @@ static int intel_lid_notify(struct notifier_block *nb, unsigned long val, | |||
| 539 | struct drm_device *dev = dev_priv->dev; | 539 | struct drm_device *dev = dev_priv->dev; |
| 540 | struct drm_connector *connector = dev_priv->int_lvds_connector; | 540 | struct drm_connector *connector = dev_priv->int_lvds_connector; |
| 541 | 541 | ||
| 542 | if (dev->switch_power_state != DRM_SWITCH_POWER_ON) | ||
| 543 | return NOTIFY_OK; | ||
| 544 | |||
| 542 | /* | 545 | /* |
| 543 | * check and update the status of LVDS connector after receiving | 546 | * check and update the status of LVDS connector after receiving |
| 544 | * the LID nofication event. | 547 | * the LID nofication event. |
diff --git a/drivers/gpu/drm/i915/intel_ringbuffer.c b/drivers/gpu/drm/i915/intel_ringbuffer.c index e9e6f71418a4..3971b5e6ad60 100644 --- a/drivers/gpu/drm/i915/intel_ringbuffer.c +++ b/drivers/gpu/drm/i915/intel_ringbuffer.c | |||
| @@ -236,7 +236,7 @@ init_pipe_control(struct intel_ring_buffer *ring) | |||
| 236 | ret = -ENOMEM; | 236 | ret = -ENOMEM; |
| 237 | goto err; | 237 | goto err; |
| 238 | } | 238 | } |
| 239 | obj->agp_type = AGP_USER_CACHED_MEMORY; | 239 | obj->cache_level = I915_CACHE_LLC; |
| 240 | 240 | ||
| 241 | ret = i915_gem_object_pin(obj, 4096, true); | 241 | ret = i915_gem_object_pin(obj, 4096, true); |
| 242 | if (ret) | 242 | if (ret) |
| @@ -286,7 +286,7 @@ static int init_render_ring(struct intel_ring_buffer *ring) | |||
| 286 | 286 | ||
| 287 | if (INTEL_INFO(dev)->gen > 3) { | 287 | if (INTEL_INFO(dev)->gen > 3) { |
| 288 | int mode = VS_TIMER_DISPATCH << 16 | VS_TIMER_DISPATCH; | 288 | int mode = VS_TIMER_DISPATCH << 16 | VS_TIMER_DISPATCH; |
| 289 | if (IS_GEN6(dev)) | 289 | if (IS_GEN6(dev) || IS_GEN7(dev)) |
| 290 | mode |= MI_FLUSH_ENABLE << 16 | MI_FLUSH_ENABLE; | 290 | mode |= MI_FLUSH_ENABLE << 16 | MI_FLUSH_ENABLE; |
| 291 | I915_WRITE(MI_MODE, mode); | 291 | I915_WRITE(MI_MODE, mode); |
| 292 | } | 292 | } |
| @@ -551,10 +551,31 @@ render_ring_put_irq(struct intel_ring_buffer *ring) | |||
| 551 | 551 | ||
| 552 | void intel_ring_setup_status_page(struct intel_ring_buffer *ring) | 552 | void intel_ring_setup_status_page(struct intel_ring_buffer *ring) |
| 553 | { | 553 | { |
| 554 | struct drm_device *dev = ring->dev; | ||
| 554 | drm_i915_private_t *dev_priv = ring->dev->dev_private; | 555 | drm_i915_private_t *dev_priv = ring->dev->dev_private; |
| 555 | u32 mmio = IS_GEN6(ring->dev) ? | 556 | u32 mmio = 0; |
| 556 | RING_HWS_PGA_GEN6(ring->mmio_base) : | 557 | |
| 557 | RING_HWS_PGA(ring->mmio_base); | 558 | /* The ring status page addresses are no longer next to the rest of |
| 559 | * the ring registers as of gen7. | ||
| 560 | */ | ||
| 561 | if (IS_GEN7(dev)) { | ||
| 562 | switch (ring->id) { | ||
| 563 | case RING_RENDER: | ||
| 564 | mmio = RENDER_HWS_PGA_GEN7; | ||
| 565 | break; | ||
| 566 | case RING_BLT: | ||
| 567 | mmio = BLT_HWS_PGA_GEN7; | ||
| 568 | break; | ||
| 569 | case RING_BSD: | ||
| 570 | mmio = BSD_HWS_PGA_GEN7; | ||
| 571 | break; | ||
| 572 | } | ||
| 573 | } else if (IS_GEN6(ring->dev)) { | ||
| 574 | mmio = RING_HWS_PGA_GEN6(ring->mmio_base); | ||
| 575 | } else { | ||
| 576 | mmio = RING_HWS_PGA(ring->mmio_base); | ||
| 577 | } | ||
| 578 | |||
| 558 | I915_WRITE(mmio, (u32)ring->status_page.gfx_addr); | 579 | I915_WRITE(mmio, (u32)ring->status_page.gfx_addr); |
| 559 | POSTING_READ(mmio); | 580 | POSTING_READ(mmio); |
| 560 | } | 581 | } |
| @@ -759,7 +780,7 @@ static int init_status_page(struct intel_ring_buffer *ring) | |||
| 759 | ret = -ENOMEM; | 780 | ret = -ENOMEM; |
| 760 | goto err; | 781 | goto err; |
| 761 | } | 782 | } |
| 762 | obj->agp_type = AGP_USER_CACHED_MEMORY; | 783 | obj->cache_level = I915_CACHE_LLC; |
| 763 | 784 | ||
| 764 | ret = i915_gem_object_pin(obj, 4096, true); | 785 | ret = i915_gem_object_pin(obj, 4096, true); |
| 765 | if (ret != 0) { | 786 | if (ret != 0) { |
| @@ -800,6 +821,7 @@ int intel_init_ring_buffer(struct drm_device *dev, | |||
| 800 | INIT_LIST_HEAD(&ring->request_list); | 821 | INIT_LIST_HEAD(&ring->request_list); |
| 801 | INIT_LIST_HEAD(&ring->gpu_write_list); | 822 | INIT_LIST_HEAD(&ring->gpu_write_list); |
| 802 | 823 | ||
| 824 | init_waitqueue_head(&ring->irq_queue); | ||
| 803 | spin_lock_init(&ring->irq_lock); | 825 | spin_lock_init(&ring->irq_lock); |
| 804 | ring->irq_mask = ~0; | 826 | ring->irq_mask = ~0; |
| 805 | 827 | ||
| @@ -872,7 +894,7 @@ void intel_cleanup_ring_buffer(struct intel_ring_buffer *ring) | |||
| 872 | 894 | ||
| 873 | /* Disable the ring buffer. The ring must be idle at this point */ | 895 | /* Disable the ring buffer. The ring must be idle at this point */ |
| 874 | dev_priv = ring->dev->dev_private; | 896 | dev_priv = ring->dev->dev_private; |
| 875 | ret = intel_wait_ring_buffer(ring, ring->size - 8); | 897 | ret = intel_wait_ring_idle(ring); |
| 876 | if (ret) | 898 | if (ret) |
| 877 | DRM_ERROR("failed to quiesce %s whilst cleaning up: %d\n", | 899 | DRM_ERROR("failed to quiesce %s whilst cleaning up: %d\n", |
| 878 | ring->name, ret); | 900 | ring->name, ret); |
| @@ -1333,7 +1355,7 @@ int intel_init_bsd_ring_buffer(struct drm_device *dev) | |||
| 1333 | drm_i915_private_t *dev_priv = dev->dev_private; | 1355 | drm_i915_private_t *dev_priv = dev->dev_private; |
| 1334 | struct intel_ring_buffer *ring = &dev_priv->ring[VCS]; | 1356 | struct intel_ring_buffer *ring = &dev_priv->ring[VCS]; |
| 1335 | 1357 | ||
| 1336 | if (IS_GEN6(dev)) | 1358 | if (IS_GEN6(dev) || IS_GEN7(dev)) |
| 1337 | *ring = gen6_bsd_ring; | 1359 | *ring = gen6_bsd_ring; |
| 1338 | else | 1360 | else |
| 1339 | *ring = bsd_ring; | 1361 | *ring = bsd_ring; |
diff --git a/drivers/gpu/drm/i915/intel_ringbuffer.h b/drivers/gpu/drm/i915/intel_ringbuffer.h index f23cc5f037a6..c0e0ee63fbf4 100644 --- a/drivers/gpu/drm/i915/intel_ringbuffer.h +++ b/drivers/gpu/drm/i915/intel_ringbuffer.h | |||
| @@ -14,27 +14,24 @@ struct intel_hw_status_page { | |||
| 14 | struct drm_i915_gem_object *obj; | 14 | struct drm_i915_gem_object *obj; |
| 15 | }; | 15 | }; |
| 16 | 16 | ||
| 17 | #define I915_RING_READ(reg) i915_gt_read(dev_priv, reg) | 17 | #define I915_READ_TAIL(ring) I915_READ(RING_TAIL((ring)->mmio_base)) |
| 18 | #define I915_RING_WRITE(reg, val) i915_gt_write(dev_priv, reg, val) | 18 | #define I915_WRITE_TAIL(ring, val) I915_WRITE(RING_TAIL((ring)->mmio_base), val) |
| 19 | 19 | ||
| 20 | #define I915_READ_TAIL(ring) I915_RING_READ(RING_TAIL((ring)->mmio_base)) | 20 | #define I915_READ_START(ring) I915_READ(RING_START((ring)->mmio_base)) |
| 21 | #define I915_WRITE_TAIL(ring, val) I915_RING_WRITE(RING_TAIL((ring)->mmio_base), val) | 21 | #define I915_WRITE_START(ring, val) I915_WRITE(RING_START((ring)->mmio_base), val) |
| 22 | 22 | ||
| 23 | #define I915_READ_START(ring) I915_RING_READ(RING_START((ring)->mmio_base)) | 23 | #define I915_READ_HEAD(ring) I915_READ(RING_HEAD((ring)->mmio_base)) |
| 24 | #define I915_WRITE_START(ring, val) I915_RING_WRITE(RING_START((ring)->mmio_base), val) | 24 | #define I915_WRITE_HEAD(ring, val) I915_WRITE(RING_HEAD((ring)->mmio_base), val) |
| 25 | 25 | ||
| 26 | #define I915_READ_HEAD(ring) I915_RING_READ(RING_HEAD((ring)->mmio_base)) | 26 | #define I915_READ_CTL(ring) I915_READ(RING_CTL((ring)->mmio_base)) |
| 27 | #define I915_WRITE_HEAD(ring, val) I915_RING_WRITE(RING_HEAD((ring)->mmio_base), val) | 27 | #define I915_WRITE_CTL(ring, val) I915_WRITE(RING_CTL((ring)->mmio_base), val) |
| 28 | 28 | ||
| 29 | #define I915_READ_CTL(ring) I915_RING_READ(RING_CTL((ring)->mmio_base)) | 29 | #define I915_READ_IMR(ring) I915_READ(RING_IMR((ring)->mmio_base)) |
| 30 | #define I915_WRITE_CTL(ring, val) I915_RING_WRITE(RING_CTL((ring)->mmio_base), val) | 30 | #define I915_WRITE_IMR(ring, val) I915_WRITE(RING_IMR((ring)->mmio_base), val) |
| 31 | 31 | ||
| 32 | #define I915_READ_IMR(ring) I915_RING_READ(RING_IMR((ring)->mmio_base)) | 32 | #define I915_READ_NOPID(ring) I915_READ(RING_NOPID((ring)->mmio_base)) |
| 33 | #define I915_WRITE_IMR(ring, val) I915_RING_WRITE(RING_IMR((ring)->mmio_base), val) | 33 | #define I915_READ_SYNC_0(ring) I915_READ(RING_SYNC_0((ring)->mmio_base)) |
| 34 | 34 | #define I915_READ_SYNC_1(ring) I915_READ(RING_SYNC_1((ring)->mmio_base)) | |
| 35 | #define I915_READ_NOPID(ring) I915_RING_READ(RING_NOPID((ring)->mmio_base)) | ||
| 36 | #define I915_READ_SYNC_0(ring) I915_RING_READ(RING_SYNC_0((ring)->mmio_base)) | ||
| 37 | #define I915_READ_SYNC_1(ring) I915_RING_READ(RING_SYNC_1((ring)->mmio_base)) | ||
| 38 | 35 | ||
| 39 | struct intel_ring_buffer { | 36 | struct intel_ring_buffer { |
| 40 | const char *name; | 37 | const char *name; |
| @@ -164,7 +161,13 @@ intel_read_status_page(struct intel_ring_buffer *ring, | |||
| 164 | #define I915_BREADCRUMB_INDEX 0x21 | 161 | #define I915_BREADCRUMB_INDEX 0x21 |
| 165 | 162 | ||
| 166 | void intel_cleanup_ring_buffer(struct intel_ring_buffer *ring); | 163 | void intel_cleanup_ring_buffer(struct intel_ring_buffer *ring); |
| 164 | |||
| 167 | int __must_check intel_wait_ring_buffer(struct intel_ring_buffer *ring, int n); | 165 | int __must_check intel_wait_ring_buffer(struct intel_ring_buffer *ring, int n); |
| 166 | static inline int intel_wait_ring_idle(struct intel_ring_buffer *ring) | ||
| 167 | { | ||
| 168 | return intel_wait_ring_buffer(ring, ring->space - 8); | ||
| 169 | } | ||
| 170 | |||
| 168 | int __must_check intel_ring_begin(struct intel_ring_buffer *ring, int n); | 171 | int __must_check intel_ring_begin(struct intel_ring_buffer *ring, int n); |
| 169 | 172 | ||
| 170 | static inline void intel_ring_emit(struct intel_ring_buffer *ring, | 173 | static inline void intel_ring_emit(struct intel_ring_buffer *ring, |
diff --git a/drivers/gpu/drm/i915/intel_tv.c b/drivers/gpu/drm/i915/intel_tv.c index 6b22c1dcc015..113e4e7264cd 100644 --- a/drivers/gpu/drm/i915/intel_tv.c +++ b/drivers/gpu/drm/i915/intel_tv.c | |||
| @@ -1361,15 +1361,14 @@ intel_tv_detect(struct drm_connector *connector, bool force) | |||
| 1361 | if (intel_tv->base.base.crtc && intel_tv->base.base.crtc->enabled) { | 1361 | if (intel_tv->base.base.crtc && intel_tv->base.base.crtc->enabled) { |
| 1362 | type = intel_tv_detect_type(intel_tv, connector); | 1362 | type = intel_tv_detect_type(intel_tv, connector); |
| 1363 | } else if (force) { | 1363 | } else if (force) { |
| 1364 | struct drm_crtc *crtc; | 1364 | struct intel_load_detect_pipe tmp; |
| 1365 | int dpms_mode; | ||
| 1366 | 1365 | ||
| 1367 | crtc = intel_get_load_detect_pipe(&intel_tv->base, connector, | 1366 | if (intel_get_load_detect_pipe(&intel_tv->base, connector, |
| 1368 | &mode, &dpms_mode); | 1367 | &mode, &tmp)) { |
| 1369 | if (crtc) { | ||
| 1370 | type = intel_tv_detect_type(intel_tv, connector); | 1368 | type = intel_tv_detect_type(intel_tv, connector); |
| 1371 | intel_release_load_detect_pipe(&intel_tv->base, connector, | 1369 | intel_release_load_detect_pipe(&intel_tv->base, |
| 1372 | dpms_mode); | 1370 | connector, |
| 1371 | &tmp); | ||
| 1373 | } else | 1372 | } else |
| 1374 | return connector_status_unknown; | 1373 | return connector_status_unknown; |
| 1375 | } else | 1374 | } else |
diff --git a/drivers/gpu/drm/radeon/evergreen.c b/drivers/gpu/drm/radeon/evergreen.c index e9bc135d9189..c20eac3379e6 100644 --- a/drivers/gpu/drm/radeon/evergreen.c +++ b/drivers/gpu/drm/radeon/evergreen.c | |||
| @@ -862,9 +862,15 @@ int evergreen_pcie_gart_enable(struct radeon_device *rdev) | |||
| 862 | SYSTEM_ACCESS_MODE_NOT_IN_SYS | | 862 | SYSTEM_ACCESS_MODE_NOT_IN_SYS | |
| 863 | SYSTEM_APERTURE_UNMAPPED_ACCESS_PASS_THRU | | 863 | SYSTEM_APERTURE_UNMAPPED_ACCESS_PASS_THRU | |
| 864 | EFFECTIVE_L1_TLB_SIZE(5) | EFFECTIVE_L1_QUEUE_SIZE(5); | 864 | EFFECTIVE_L1_TLB_SIZE(5) | EFFECTIVE_L1_QUEUE_SIZE(5); |
| 865 | WREG32(MC_VM_MD_L1_TLB0_CNTL, tmp); | 865 | if (rdev->flags & RADEON_IS_IGP) { |
| 866 | WREG32(MC_VM_MD_L1_TLB1_CNTL, tmp); | 866 | WREG32(FUS_MC_VM_MD_L1_TLB0_CNTL, tmp); |
| 867 | WREG32(MC_VM_MD_L1_TLB2_CNTL, tmp); | 867 | WREG32(FUS_MC_VM_MD_L1_TLB1_CNTL, tmp); |
| 868 | WREG32(FUS_MC_VM_MD_L1_TLB2_CNTL, tmp); | ||
| 869 | } else { | ||
| 870 | WREG32(MC_VM_MD_L1_TLB0_CNTL, tmp); | ||
| 871 | WREG32(MC_VM_MD_L1_TLB1_CNTL, tmp); | ||
| 872 | WREG32(MC_VM_MD_L1_TLB2_CNTL, tmp); | ||
| 873 | } | ||
| 868 | WREG32(MC_VM_MB_L1_TLB0_CNTL, tmp); | 874 | WREG32(MC_VM_MB_L1_TLB0_CNTL, tmp); |
| 869 | WREG32(MC_VM_MB_L1_TLB1_CNTL, tmp); | 875 | WREG32(MC_VM_MB_L1_TLB1_CNTL, tmp); |
| 870 | WREG32(MC_VM_MB_L1_TLB2_CNTL, tmp); | 876 | WREG32(MC_VM_MB_L1_TLB2_CNTL, tmp); |
| @@ -2923,11 +2929,6 @@ static int evergreen_startup(struct radeon_device *rdev) | |||
| 2923 | rdev->asic->copy = NULL; | 2929 | rdev->asic->copy = NULL; |
| 2924 | dev_warn(rdev->dev, "failed blitter (%d) falling back to memcpy\n", r); | 2930 | dev_warn(rdev->dev, "failed blitter (%d) falling back to memcpy\n", r); |
| 2925 | } | 2931 | } |
| 2926 | /* XXX: ontario has problems blitting to gart at the moment */ | ||
| 2927 | if (rdev->family == CHIP_PALM) { | ||
| 2928 | rdev->asic->copy = NULL; | ||
| 2929 | radeon_ttm_set_active_vram_size(rdev, rdev->mc.visible_vram_size); | ||
| 2930 | } | ||
| 2931 | 2932 | ||
| 2932 | /* allocate wb buffer */ | 2933 | /* allocate wb buffer */ |
| 2933 | r = radeon_wb_init(rdev); | 2934 | r = radeon_wb_init(rdev); |
diff --git a/drivers/gpu/drm/radeon/evergreend.h b/drivers/gpu/drm/radeon/evergreend.h index 9aaa3f0c9372..94533849927e 100644 --- a/drivers/gpu/drm/radeon/evergreend.h +++ b/drivers/gpu/drm/radeon/evergreend.h | |||
| @@ -221,6 +221,11 @@ | |||
| 221 | #define MC_VM_MD_L1_TLB0_CNTL 0x2654 | 221 | #define MC_VM_MD_L1_TLB0_CNTL 0x2654 |
| 222 | #define MC_VM_MD_L1_TLB1_CNTL 0x2658 | 222 | #define MC_VM_MD_L1_TLB1_CNTL 0x2658 |
| 223 | #define MC_VM_MD_L1_TLB2_CNTL 0x265C | 223 | #define MC_VM_MD_L1_TLB2_CNTL 0x265C |
| 224 | |||
| 225 | #define FUS_MC_VM_MD_L1_TLB0_CNTL 0x265C | ||
| 226 | #define FUS_MC_VM_MD_L1_TLB1_CNTL 0x2660 | ||
| 227 | #define FUS_MC_VM_MD_L1_TLB2_CNTL 0x2664 | ||
| 228 | |||
| 224 | #define MC_VM_SYSTEM_APERTURE_DEFAULT_ADDR 0x203C | 229 | #define MC_VM_SYSTEM_APERTURE_DEFAULT_ADDR 0x203C |
| 225 | #define MC_VM_SYSTEM_APERTURE_HIGH_ADDR 0x2038 | 230 | #define MC_VM_SYSTEM_APERTURE_HIGH_ADDR 0x2038 |
| 226 | #define MC_VM_SYSTEM_APERTURE_LOW_ADDR 0x2034 | 231 | #define MC_VM_SYSTEM_APERTURE_LOW_ADDR 0x2034 |
diff --git a/drivers/gpu/drm/radeon/radeon_atombios.c b/drivers/gpu/drm/radeon/radeon_atombios.c index f5d12fb103fa..f116516bfef7 100644 --- a/drivers/gpu/drm/radeon/radeon_atombios.c +++ b/drivers/gpu/drm/radeon/radeon_atombios.c | |||
| @@ -1599,9 +1599,10 @@ struct radeon_encoder_atom_dig *radeon_atombios_get_lvds_info(struct | |||
| 1599 | memcpy((u8 *)edid, (u8 *)&fake_edid_record->ucFakeEDIDString[0], | 1599 | memcpy((u8 *)edid, (u8 *)&fake_edid_record->ucFakeEDIDString[0], |
| 1600 | fake_edid_record->ucFakeEDIDLength); | 1600 | fake_edid_record->ucFakeEDIDLength); |
| 1601 | 1601 | ||
| 1602 | if (drm_edid_is_valid(edid)) | 1602 | if (drm_edid_is_valid(edid)) { |
| 1603 | rdev->mode_info.bios_hardcoded_edid = edid; | 1603 | rdev->mode_info.bios_hardcoded_edid = edid; |
| 1604 | else | 1604 | rdev->mode_info.bios_hardcoded_edid_size = edid_size; |
| 1605 | } else | ||
| 1605 | kfree(edid); | 1606 | kfree(edid); |
| 1606 | } | 1607 | } |
| 1607 | } | 1608 | } |
diff --git a/drivers/gpu/drm/radeon/radeon_kms.c b/drivers/gpu/drm/radeon/radeon_kms.c index 871df0376b1c..bd58af658581 100644 --- a/drivers/gpu/drm/radeon/radeon_kms.c +++ b/drivers/gpu/drm/radeon/radeon_kms.c | |||
| @@ -234,6 +234,9 @@ int radeon_info_ioctl(struct drm_device *dev, void *data, struct drm_file *filp) | |||
| 234 | return -EINVAL; | 234 | return -EINVAL; |
| 235 | } | 235 | } |
| 236 | break; | 236 | break; |
| 237 | case RADEON_INFO_FUSION_GART_WORKING: | ||
| 238 | value = 1; | ||
| 239 | break; | ||
| 237 | default: | 240 | default: |
| 238 | DRM_DEBUG_KMS("Invalid request %d\n", info->request); | 241 | DRM_DEBUG_KMS("Invalid request %d\n", info->request); |
| 239 | return -EINVAL; | 242 | return -EINVAL; |
diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig index 060ef6327876..50e40dbd8bb6 100644 --- a/drivers/hwmon/Kconfig +++ b/drivers/hwmon/Kconfig | |||
| @@ -110,8 +110,7 @@ config SENSORS_ADM1021 | |||
| 110 | help | 110 | help |
| 111 | If you say yes here you get support for Analog Devices ADM1021 | 111 | If you say yes here you get support for Analog Devices ADM1021 |
| 112 | and ADM1023 sensor chips and clones: Maxim MAX1617 and MAX1617A, | 112 | and ADM1023 sensor chips and clones: Maxim MAX1617 and MAX1617A, |
| 113 | Genesys Logic GL523SM, National Semiconductor LM84, TI THMC10, | 113 | Genesys Logic GL523SM, National Semiconductor LM84 and TI THMC10. |
| 114 | and the XEON processor built-in sensor. | ||
| 115 | 114 | ||
| 116 | This driver can also be built as a module. If so, the module | 115 | This driver can also be built as a module. If so, the module |
| 117 | will be called adm1021. | 116 | will be called adm1021. |
| @@ -618,10 +617,10 @@ config SENSORS_LM90 | |||
| 618 | depends on I2C | 617 | depends on I2C |
| 619 | help | 618 | help |
| 620 | If you say yes here you get support for National Semiconductor LM90, | 619 | If you say yes here you get support for National Semiconductor LM90, |
| 621 | LM86, LM89 and LM99, Analog Devices ADM1032 and ADT7461, Maxim | 620 | LM86, LM89 and LM99, Analog Devices ADM1032, ADT7461, and ADT7461A, |
| 622 | MAX6646, MAX6647, MAX6648, MAX6649, MAX6657, MAX6658, MAX6659, | 621 | Maxim MAX6646, MAX6647, MAX6648, MAX6649, MAX6657, MAX6658, MAX6659, |
| 623 | MAX6680, MAX6681, MAX6692, MAX6695, MAX6696, and Winbond/Nuvoton | 622 | MAX6680, MAX6681, MAX6692, MAX6695, MAX6696, ON Semiconductor NCT1008, |
| 624 | W83L771W/G/AWG/ASG sensor chips. | 623 | and Winbond/Nuvoton W83L771W/G/AWG/ASG sensor chips. |
| 625 | 624 | ||
| 626 | This driver can also be built as a module. If so, the module | 625 | This driver can also be built as a module. If so, the module |
| 627 | will be called lm90. | 626 | will be called lm90. |
diff --git a/drivers/hwmon/lm85.c b/drivers/hwmon/lm85.c index 250d099ca398..da72dc12068c 100644 --- a/drivers/hwmon/lm85.c +++ b/drivers/hwmon/lm85.c | |||
| @@ -1094,6 +1094,7 @@ static struct attribute *lm85_attributes_minctl[] = { | |||
| 1094 | &sensor_dev_attr_pwm1_auto_pwm_minctl.dev_attr.attr, | 1094 | &sensor_dev_attr_pwm1_auto_pwm_minctl.dev_attr.attr, |
| 1095 | &sensor_dev_attr_pwm2_auto_pwm_minctl.dev_attr.attr, | 1095 | &sensor_dev_attr_pwm2_auto_pwm_minctl.dev_attr.attr, |
| 1096 | &sensor_dev_attr_pwm3_auto_pwm_minctl.dev_attr.attr, | 1096 | &sensor_dev_attr_pwm3_auto_pwm_minctl.dev_attr.attr, |
| 1097 | NULL | ||
| 1097 | }; | 1098 | }; |
| 1098 | 1099 | ||
| 1099 | static const struct attribute_group lm85_group_minctl = { | 1100 | static const struct attribute_group lm85_group_minctl = { |
| @@ -1104,6 +1105,7 @@ static struct attribute *lm85_attributes_temp_off[] = { | |||
| 1104 | &sensor_dev_attr_temp1_auto_temp_off.dev_attr.attr, | 1105 | &sensor_dev_attr_temp1_auto_temp_off.dev_attr.attr, |
| 1105 | &sensor_dev_attr_temp2_auto_temp_off.dev_attr.attr, | 1106 | &sensor_dev_attr_temp2_auto_temp_off.dev_attr.attr, |
| 1106 | &sensor_dev_attr_temp3_auto_temp_off.dev_attr.attr, | 1107 | &sensor_dev_attr_temp3_auto_temp_off.dev_attr.attr, |
| 1108 | NULL | ||
| 1107 | }; | 1109 | }; |
| 1108 | 1110 | ||
| 1109 | static const struct attribute_group lm85_group_temp_off = { | 1111 | static const struct attribute_group lm85_group_temp_off = { |
| @@ -1329,11 +1331,11 @@ static int lm85_probe(struct i2c_client *client, | |||
| 1329 | if (data->type != emc6d103s) { | 1331 | if (data->type != emc6d103s) { |
| 1330 | err = sysfs_create_group(&client->dev.kobj, &lm85_group_minctl); | 1332 | err = sysfs_create_group(&client->dev.kobj, &lm85_group_minctl); |
| 1331 | if (err) | 1333 | if (err) |
| 1332 | goto err_kfree; | 1334 | goto err_remove_files; |
| 1333 | err = sysfs_create_group(&client->dev.kobj, | 1335 | err = sysfs_create_group(&client->dev.kobj, |
| 1334 | &lm85_group_temp_off); | 1336 | &lm85_group_temp_off); |
| 1335 | if (err) | 1337 | if (err) |
| 1336 | goto err_kfree; | 1338 | goto err_remove_files; |
| 1337 | } | 1339 | } |
| 1338 | 1340 | ||
| 1339 | /* The ADT7463/68 have an optional VRM 10 mode where pin 21 is used | 1341 | /* The ADT7463/68 have an optional VRM 10 mode where pin 21 is used |
diff --git a/drivers/hwmon/lm90.c b/drivers/hwmon/lm90.c index c43b4e9f96a9..2f94f9504804 100644 --- a/drivers/hwmon/lm90.c +++ b/drivers/hwmon/lm90.c | |||
| @@ -49,10 +49,10 @@ | |||
| 49 | * chips, but support three temperature sensors instead of two. MAX6695 | 49 | * chips, but support three temperature sensors instead of two. MAX6695 |
| 50 | * and MAX6696 only differ in the pinout so they can be treated identically. | 50 | * and MAX6696 only differ in the pinout so they can be treated identically. |
| 51 | * | 51 | * |
| 52 | * This driver also supports the ADT7461 chip from Analog Devices. | 52 | * This driver also supports ADT7461 and ADT7461A from Analog Devices as well as |
| 53 | * It's supported in both compatibility and extended mode. It is mostly | 53 | * NCT1008 from ON Semiconductor. The chips are supported in both compatibility |
| 54 | * compatible with LM90 except for a data format difference for the | 54 | * and extended mode. They are mostly compatible with LM90 except for a data |
| 55 | * temperature value registers. | 55 | * format difference for the temperature value registers. |
| 56 | * | 56 | * |
| 57 | * Since the LM90 was the first chipset supported by this driver, most | 57 | * Since the LM90 was the first chipset supported by this driver, most |
| 58 | * comments will refer to this chipset, but are actually general and | 58 | * comments will refer to this chipset, but are actually general and |
| @@ -88,9 +88,10 @@ | |||
| 88 | * Addresses to scan | 88 | * Addresses to scan |
| 89 | * Address is fully defined internally and cannot be changed except for | 89 | * Address is fully defined internally and cannot be changed except for |
| 90 | * MAX6659, MAX6680 and MAX6681. | 90 | * MAX6659, MAX6680 and MAX6681. |
| 91 | * LM86, LM89, LM90, LM99, ADM1032, ADM1032-1, ADT7461, MAX6649, MAX6657, | 91 | * LM86, LM89, LM90, LM99, ADM1032, ADM1032-1, ADT7461, ADT7461A, MAX6649, |
| 92 | * MAX6658 and W83L771 have address 0x4c. | 92 | * MAX6657, MAX6658, NCT1008 and W83L771 have address 0x4c. |
| 93 | * ADM1032-2, ADT7461-2, LM89-1, LM99-1 and MAX6646 have address 0x4d. | 93 | * ADM1032-2, ADT7461-2, ADT7461A-2, LM89-1, LM99-1, MAX6646, and NCT1008D |
| 94 | * have address 0x4d. | ||
| 94 | * MAX6647 has address 0x4e. | 95 | * MAX6647 has address 0x4e. |
| 95 | * MAX6659 can have address 0x4c, 0x4d or 0x4e. | 96 | * MAX6659 can have address 0x4c, 0x4d or 0x4e. |
| 96 | * MAX6680 and MAX6681 can have address 0x18, 0x19, 0x1a, 0x29, 0x2a, 0x2b, | 97 | * MAX6680 and MAX6681 can have address 0x18, 0x19, 0x1a, 0x29, 0x2a, 0x2b, |
| @@ -174,6 +175,7 @@ enum chips { lm90, adm1032, lm99, lm86, max6657, max6659, adt7461, max6680, | |||
| 174 | static const struct i2c_device_id lm90_id[] = { | 175 | static const struct i2c_device_id lm90_id[] = { |
| 175 | { "adm1032", adm1032 }, | 176 | { "adm1032", adm1032 }, |
| 176 | { "adt7461", adt7461 }, | 177 | { "adt7461", adt7461 }, |
| 178 | { "adt7461a", adt7461 }, | ||
| 177 | { "lm90", lm90 }, | 179 | { "lm90", lm90 }, |
| 178 | { "lm86", lm86 }, | 180 | { "lm86", lm86 }, |
| 179 | { "lm89", lm86 }, | 181 | { "lm89", lm86 }, |
| @@ -188,6 +190,7 @@ static const struct i2c_device_id lm90_id[] = { | |||
| 188 | { "max6681", max6680 }, | 190 | { "max6681", max6680 }, |
| 189 | { "max6695", max6696 }, | 191 | { "max6695", max6696 }, |
| 190 | { "max6696", max6696 }, | 192 | { "max6696", max6696 }, |
| 193 | { "nct1008", adt7461 }, | ||
| 191 | { "w83l771", w83l771 }, | 194 | { "w83l771", w83l771 }, |
| 192 | { } | 195 | { } |
| 193 | }; | 196 | }; |
| @@ -1153,6 +1156,11 @@ static int lm90_detect(struct i2c_client *new_client, | |||
| 1153 | && (reg_config1 & 0x1B) == 0x00 | 1156 | && (reg_config1 & 0x1B) == 0x00 |
| 1154 | && reg_convrate <= 0x0A) { | 1157 | && reg_convrate <= 0x0A) { |
| 1155 | name = "adt7461"; | 1158 | name = "adt7461"; |
| 1159 | } else | ||
| 1160 | if (chip_id == 0x57 /* ADT7461A, NCT1008 */ | ||
| 1161 | && (reg_config1 & 0x1B) == 0x00 | ||
| 1162 | && reg_convrate <= 0x0A) { | ||
| 1163 | name = "adt7461a"; | ||
| 1156 | } | 1164 | } |
| 1157 | } else | 1165 | } else |
| 1158 | if (man_id == 0x4D) { /* Maxim */ | 1166 | if (man_id == 0x4D) { /* Maxim */ |
diff --git a/drivers/hwmon/twl4030-madc-hwmon.c b/drivers/hwmon/twl4030-madc-hwmon.c index de5819199e2e..57240740b161 100644 --- a/drivers/hwmon/twl4030-madc-hwmon.c +++ b/drivers/hwmon/twl4030-madc-hwmon.c | |||
| @@ -98,7 +98,6 @@ static const struct attribute_group twl4030_madc_group = { | |||
| 98 | static int __devinit twl4030_madc_hwmon_probe(struct platform_device *pdev) | 98 | static int __devinit twl4030_madc_hwmon_probe(struct platform_device *pdev) |
| 99 | { | 99 | { |
| 100 | int ret; | 100 | int ret; |
| 101 | int status; | ||
| 102 | struct device *hwmon; | 101 | struct device *hwmon; |
| 103 | 102 | ||
| 104 | ret = sysfs_create_group(&pdev->dev.kobj, &twl4030_madc_group); | 103 | ret = sysfs_create_group(&pdev->dev.kobj, &twl4030_madc_group); |
| @@ -107,7 +106,7 @@ static int __devinit twl4030_madc_hwmon_probe(struct platform_device *pdev) | |||
| 107 | hwmon = hwmon_device_register(&pdev->dev); | 106 | hwmon = hwmon_device_register(&pdev->dev); |
| 108 | if (IS_ERR(hwmon)) { | 107 | if (IS_ERR(hwmon)) { |
| 109 | dev_err(&pdev->dev, "hwmon_device_register failed.\n"); | 108 | dev_err(&pdev->dev, "hwmon_device_register failed.\n"); |
| 110 | status = PTR_ERR(hwmon); | 109 | ret = PTR_ERR(hwmon); |
| 111 | goto err_reg; | 110 | goto err_reg; |
| 112 | } | 111 | } |
| 113 | 112 | ||
diff --git a/drivers/i2c/busses/i2c-i801.c b/drivers/i2c/busses/i2c-i801.c index 72c0415f6f94..455e909bc768 100644 --- a/drivers/i2c/busses/i2c-i801.c +++ b/drivers/i2c/busses/i2c-i801.c | |||
| @@ -134,10 +134,15 @@ | |||
| 134 | SMBHSTSTS_BUS_ERR | SMBHSTSTS_DEV_ERR | \ | 134 | SMBHSTSTS_BUS_ERR | SMBHSTSTS_DEV_ERR | \ |
| 135 | SMBHSTSTS_INTR) | 135 | SMBHSTSTS_INTR) |
| 136 | 136 | ||
| 137 | /* Older devices have their ID defined in <linux/pci_ids.h> */ | ||
| 138 | #define PCI_DEVICE_ID_INTEL_COUGARPOINT_SMBUS 0x1c22 | ||
| 139 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS 0x1d22 | ||
| 137 | /* Patsburg also has three 'Integrated Device Function' SMBus controllers */ | 140 | /* Patsburg also has three 'Integrated Device Function' SMBus controllers */ |
| 138 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF0 0x1d70 | 141 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF0 0x1d70 |
| 139 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF1 0x1d71 | 142 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF1 0x1d71 |
| 140 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF2 0x1d72 | 143 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF2 0x1d72 |
| 144 | #define PCI_DEVICE_ID_INTEL_DH89XXCC_SMBUS 0x2330 | ||
| 145 | #define PCI_DEVICE_ID_INTEL_5_3400_SERIES_SMBUS 0x3b30 | ||
| 141 | 146 | ||
| 142 | struct i801_priv { | 147 | struct i801_priv { |
| 143 | struct i2c_adapter adapter; | 148 | struct i2c_adapter adapter; |
diff --git a/drivers/i2c/busses/i2c-parport.c b/drivers/i2c/busses/i2c-parport.c index 0eb1515541e7..2dbba163b102 100644 --- a/drivers/i2c/busses/i2c-parport.c +++ b/drivers/i2c/busses/i2c-parport.c | |||
| @@ -1,7 +1,7 @@ | |||
| 1 | /* ------------------------------------------------------------------------ * | 1 | /* ------------------------------------------------------------------------ * |
| 2 | * i2c-parport.c I2C bus over parallel port * | 2 | * i2c-parport.c I2C bus over parallel port * |
| 3 | * ------------------------------------------------------------------------ * | 3 | * ------------------------------------------------------------------------ * |
| 4 | Copyright (C) 2003-2010 Jean Delvare <khali@linux-fr.org> | 4 | Copyright (C) 2003-2011 Jean Delvare <khali@linux-fr.org> |
| 5 | 5 | ||
| 6 | Based on older i2c-philips-par.c driver | 6 | Based on older i2c-philips-par.c driver |
| 7 | Copyright (C) 1995-2000 Simon G. Vogl | 7 | Copyright (C) 1995-2000 Simon G. Vogl |
| @@ -33,6 +33,8 @@ | |||
| 33 | #include <linux/i2c-algo-bit.h> | 33 | #include <linux/i2c-algo-bit.h> |
| 34 | #include <linux/i2c-smbus.h> | 34 | #include <linux/i2c-smbus.h> |
| 35 | #include <linux/slab.h> | 35 | #include <linux/slab.h> |
| 36 | #include <linux/list.h> | ||
| 37 | #include <linux/mutex.h> | ||
| 36 | #include "i2c-parport.h" | 38 | #include "i2c-parport.h" |
| 37 | 39 | ||
| 38 | /* ----- Device list ------------------------------------------------------ */ | 40 | /* ----- Device list ------------------------------------------------------ */ |
| @@ -43,10 +45,11 @@ struct i2c_par { | |||
| 43 | struct i2c_algo_bit_data algo_data; | 45 | struct i2c_algo_bit_data algo_data; |
| 44 | struct i2c_smbus_alert_setup alert_data; | 46 | struct i2c_smbus_alert_setup alert_data; |
| 45 | struct i2c_client *ara; | 47 | struct i2c_client *ara; |
| 46 | struct i2c_par *next; | 48 | struct list_head node; |
| 47 | }; | 49 | }; |
| 48 | 50 | ||
| 49 | static struct i2c_par *adapter_list; | 51 | static LIST_HEAD(adapter_list); |
| 52 | static DEFINE_MUTEX(adapter_list_lock); | ||
| 50 | 53 | ||
| 51 | /* ----- Low-level parallel port access ----------------------------------- */ | 54 | /* ----- Low-level parallel port access ----------------------------------- */ |
| 52 | 55 | ||
| @@ -228,8 +231,9 @@ static void i2c_parport_attach (struct parport *port) | |||
| 228 | } | 231 | } |
| 229 | 232 | ||
| 230 | /* Add the new adapter to the list */ | 233 | /* Add the new adapter to the list */ |
| 231 | adapter->next = adapter_list; | 234 | mutex_lock(&adapter_list_lock); |
| 232 | adapter_list = adapter; | 235 | list_add_tail(&adapter->node, &adapter_list); |
| 236 | mutex_unlock(&adapter_list_lock); | ||
| 233 | return; | 237 | return; |
| 234 | 238 | ||
| 235 | ERROR1: | 239 | ERROR1: |
| @@ -241,11 +245,11 @@ ERROR0: | |||
| 241 | 245 | ||
| 242 | static void i2c_parport_detach (struct parport *port) | 246 | static void i2c_parport_detach (struct parport *port) |
| 243 | { | 247 | { |
| 244 | struct i2c_par *adapter, *prev; | 248 | struct i2c_par *adapter, *_n; |
| 245 | 249 | ||
| 246 | /* Walk the list */ | 250 | /* Walk the list */ |
| 247 | for (prev = NULL, adapter = adapter_list; adapter; | 251 | mutex_lock(&adapter_list_lock); |
| 248 | prev = adapter, adapter = adapter->next) { | 252 | list_for_each_entry_safe(adapter, _n, &adapter_list, node) { |
| 249 | if (adapter->pdev->port == port) { | 253 | if (adapter->pdev->port == port) { |
| 250 | if (adapter->ara) { | 254 | if (adapter->ara) { |
| 251 | parport_disable_irq(port); | 255 | parport_disable_irq(port); |
| @@ -259,14 +263,11 @@ static void i2c_parport_detach (struct parport *port) | |||
| 259 | 263 | ||
| 260 | parport_release(adapter->pdev); | 264 | parport_release(adapter->pdev); |
| 261 | parport_unregister_device(adapter->pdev); | 265 | parport_unregister_device(adapter->pdev); |
| 262 | if (prev) | 266 | list_del(&adapter->node); |
| 263 | prev->next = adapter->next; | ||
| 264 | else | ||
| 265 | adapter_list = adapter->next; | ||
| 266 | kfree(adapter); | 267 | kfree(adapter); |
| 267 | return; | ||
| 268 | } | 268 | } |
| 269 | } | 269 | } |
| 270 | mutex_unlock(&adapter_list_lock); | ||
| 270 | } | 271 | } |
| 271 | 272 | ||
| 272 | static struct parport_driver i2c_parport_driver = { | 273 | static struct parport_driver i2c_parport_driver = { |
diff --git a/drivers/infiniband/hw/qib/qib_iba6120.c b/drivers/infiniband/hw/qib/qib_iba6120.c index 7de4b7ebffc5..d8ca0a0b970d 100644 --- a/drivers/infiniband/hw/qib/qib_iba6120.c +++ b/drivers/infiniband/hw/qib/qib_iba6120.c | |||
| @@ -1799,7 +1799,7 @@ static int qib_6120_setup_reset(struct qib_devdata *dd) | |||
| 1799 | /* | 1799 | /* |
| 1800 | * Keep chip from being accessed until we are ready. Use | 1800 | * Keep chip from being accessed until we are ready. Use |
| 1801 | * writeq() directly, to allow the write even though QIB_PRESENT | 1801 | * writeq() directly, to allow the write even though QIB_PRESENT |
| 1802 | * isn't' set. | 1802 | * isn't set. |
| 1803 | */ | 1803 | */ |
| 1804 | dd->flags &= ~(QIB_INITTED | QIB_PRESENT); | 1804 | dd->flags &= ~(QIB_INITTED | QIB_PRESENT); |
| 1805 | dd->int_counter = 0; /* so we check interrupts work again */ | 1805 | dd->int_counter = 0; /* so we check interrupts work again */ |
diff --git a/drivers/infiniband/hw/qib/qib_iba7220.c b/drivers/infiniband/hw/qib/qib_iba7220.c index 74fe0360bec7..c765a2eb04cf 100644 --- a/drivers/infiniband/hw/qib/qib_iba7220.c +++ b/drivers/infiniband/hw/qib/qib_iba7220.c | |||
| @@ -2111,7 +2111,7 @@ static int qib_setup_7220_reset(struct qib_devdata *dd) | |||
| 2111 | /* | 2111 | /* |
| 2112 | * Keep chip from being accessed until we are ready. Use | 2112 | * Keep chip from being accessed until we are ready. Use |
| 2113 | * writeq() directly, to allow the write even though QIB_PRESENT | 2113 | * writeq() directly, to allow the write even though QIB_PRESENT |
| 2114 | * isn't' set. | 2114 | * isn't set. |
| 2115 | */ | 2115 | */ |
| 2116 | dd->flags &= ~(QIB_INITTED | QIB_PRESENT); | 2116 | dd->flags &= ~(QIB_INITTED | QIB_PRESENT); |
| 2117 | dd->int_counter = 0; /* so we check interrupts work again */ | 2117 | dd->int_counter = 0; /* so we check interrupts work again */ |
diff --git a/drivers/infiniband/hw/qib/qib_iba7322.c b/drivers/infiniband/hw/qib/qib_iba7322.c index 55de3cf3441c..6bab3eaea70f 100644 --- a/drivers/infiniband/hw/qib/qib_iba7322.c +++ b/drivers/infiniband/hw/qib/qib_iba7322.c | |||
| @@ -3299,7 +3299,7 @@ static int qib_do_7322_reset(struct qib_devdata *dd) | |||
| 3299 | /* | 3299 | /* |
| 3300 | * Keep chip from being accessed until we are ready. Use | 3300 | * Keep chip from being accessed until we are ready. Use |
| 3301 | * writeq() directly, to allow the write even though QIB_PRESENT | 3301 | * writeq() directly, to allow the write even though QIB_PRESENT |
| 3302 | * isn't' set. | 3302 | * isn't set. |
| 3303 | */ | 3303 | */ |
| 3304 | dd->flags &= ~(QIB_INITTED | QIB_PRESENT | QIB_BADINTR); | 3304 | dd->flags &= ~(QIB_INITTED | QIB_PRESENT | QIB_BADINTR); |
| 3305 | dd->flags |= QIB_DOING_RESET; | 3305 | dd->flags |= QIB_DOING_RESET; |
diff --git a/drivers/input/touchscreen/wm831x-ts.c b/drivers/input/touchscreen/wm831x-ts.c index 6ae054f8e0aa..9175d49d2546 100644 --- a/drivers/input/touchscreen/wm831x-ts.c +++ b/drivers/input/touchscreen/wm831x-ts.c | |||
| @@ -68,8 +68,23 @@ struct wm831x_ts { | |||
| 68 | unsigned int pd_irq; | 68 | unsigned int pd_irq; |
| 69 | bool pressure; | 69 | bool pressure; |
| 70 | bool pen_down; | 70 | bool pen_down; |
| 71 | struct work_struct pd_data_work; | ||
| 71 | }; | 72 | }; |
| 72 | 73 | ||
| 74 | static void wm831x_pd_data_work(struct work_struct *work) | ||
| 75 | { | ||
| 76 | struct wm831x_ts *wm831x_ts = | ||
| 77 | container_of(work, struct wm831x_ts, pd_data_work); | ||
| 78 | |||
| 79 | if (wm831x_ts->pen_down) { | ||
| 80 | enable_irq(wm831x_ts->data_irq); | ||
| 81 | dev_dbg(wm831x_ts->wm831x->dev, "IRQ PD->DATA done\n"); | ||
| 82 | } else { | ||
| 83 | enable_irq(wm831x_ts->pd_irq); | ||
| 84 | dev_dbg(wm831x_ts->wm831x->dev, "IRQ DATA->PD done\n"); | ||
| 85 | } | ||
| 86 | } | ||
| 87 | |||
| 73 | static irqreturn_t wm831x_ts_data_irq(int irq, void *irq_data) | 88 | static irqreturn_t wm831x_ts_data_irq(int irq, void *irq_data) |
| 74 | { | 89 | { |
| 75 | struct wm831x_ts *wm831x_ts = irq_data; | 90 | struct wm831x_ts *wm831x_ts = irq_data; |
| @@ -110,6 +125,9 @@ static irqreturn_t wm831x_ts_data_irq(int irq, void *irq_data) | |||
| 110 | } | 125 | } |
| 111 | 126 | ||
| 112 | if (!wm831x_ts->pen_down) { | 127 | if (!wm831x_ts->pen_down) { |
| 128 | /* Switch from data to pen down */ | ||
| 129 | dev_dbg(wm831x->dev, "IRQ DATA->PD\n"); | ||
| 130 | |||
| 113 | disable_irq_nosync(wm831x_ts->data_irq); | 131 | disable_irq_nosync(wm831x_ts->data_irq); |
| 114 | 132 | ||
| 115 | /* Don't need data any more */ | 133 | /* Don't need data any more */ |
| @@ -128,6 +146,10 @@ static irqreturn_t wm831x_ts_data_irq(int irq, void *irq_data) | |||
| 128 | ABS_PRESSURE, 0); | 146 | ABS_PRESSURE, 0); |
| 129 | 147 | ||
| 130 | input_report_key(wm831x_ts->input_dev, BTN_TOUCH, 0); | 148 | input_report_key(wm831x_ts->input_dev, BTN_TOUCH, 0); |
| 149 | |||
| 150 | schedule_work(&wm831x_ts->pd_data_work); | ||
| 151 | } else { | ||
| 152 | input_report_key(wm831x_ts->input_dev, BTN_TOUCH, 1); | ||
| 131 | } | 153 | } |
| 132 | 154 | ||
| 133 | input_sync(wm831x_ts->input_dev); | 155 | input_sync(wm831x_ts->input_dev); |
| @@ -141,6 +163,11 @@ static irqreturn_t wm831x_ts_pen_down_irq(int irq, void *irq_data) | |||
| 141 | struct wm831x *wm831x = wm831x_ts->wm831x; | 163 | struct wm831x *wm831x = wm831x_ts->wm831x; |
| 142 | int ena = 0; | 164 | int ena = 0; |
| 143 | 165 | ||
| 166 | if (wm831x_ts->pen_down) | ||
| 167 | return IRQ_HANDLED; | ||
| 168 | |||
| 169 | disable_irq_nosync(wm831x_ts->pd_irq); | ||
| 170 | |||
| 144 | /* Start collecting data */ | 171 | /* Start collecting data */ |
| 145 | if (wm831x_ts->pressure) | 172 | if (wm831x_ts->pressure) |
| 146 | ena |= WM831X_TCH_Z_ENA; | 173 | ena |= WM831X_TCH_Z_ENA; |
| @@ -149,14 +176,14 @@ static irqreturn_t wm831x_ts_pen_down_irq(int irq, void *irq_data) | |||
| 149 | WM831X_TCH_X_ENA | WM831X_TCH_Y_ENA | WM831X_TCH_Z_ENA, | 176 | WM831X_TCH_X_ENA | WM831X_TCH_Y_ENA | WM831X_TCH_Z_ENA, |
| 150 | WM831X_TCH_X_ENA | WM831X_TCH_Y_ENA | ena); | 177 | WM831X_TCH_X_ENA | WM831X_TCH_Y_ENA | ena); |
| 151 | 178 | ||
| 152 | input_report_key(wm831x_ts->input_dev, BTN_TOUCH, 1); | ||
| 153 | input_sync(wm831x_ts->input_dev); | ||
| 154 | |||
| 155 | wm831x_set_bits(wm831x, WM831X_INTERRUPT_STATUS_1, | 179 | wm831x_set_bits(wm831x, WM831X_INTERRUPT_STATUS_1, |
| 156 | WM831X_TCHPD_EINT, WM831X_TCHPD_EINT); | 180 | WM831X_TCHPD_EINT, WM831X_TCHPD_EINT); |
| 157 | 181 | ||
| 158 | wm831x_ts->pen_down = true; | 182 | wm831x_ts->pen_down = true; |
| 159 | enable_irq(wm831x_ts->data_irq); | 183 | |
| 184 | /* Switch from pen down to data */ | ||
| 185 | dev_dbg(wm831x->dev, "IRQ PD->DATA\n"); | ||
| 186 | schedule_work(&wm831x_ts->pd_data_work); | ||
| 160 | 187 | ||
| 161 | return IRQ_HANDLED; | 188 | return IRQ_HANDLED; |
| 162 | } | 189 | } |
| @@ -182,13 +209,28 @@ static void wm831x_ts_input_close(struct input_dev *idev) | |||
| 182 | struct wm831x_ts *wm831x_ts = input_get_drvdata(idev); | 209 | struct wm831x_ts *wm831x_ts = input_get_drvdata(idev); |
| 183 | struct wm831x *wm831x = wm831x_ts->wm831x; | 210 | struct wm831x *wm831x = wm831x_ts->wm831x; |
| 184 | 211 | ||
| 212 | /* Shut the controller down, disabling all other functionality too */ | ||
| 185 | wm831x_set_bits(wm831x, WM831X_TOUCH_CONTROL_1, | 213 | wm831x_set_bits(wm831x, WM831X_TOUCH_CONTROL_1, |
| 186 | WM831X_TCH_ENA | WM831X_TCH_CVT_ENA | | 214 | WM831X_TCH_ENA | WM831X_TCH_X_ENA | |
| 187 | WM831X_TCH_X_ENA | WM831X_TCH_Y_ENA | | 215 | WM831X_TCH_Y_ENA | WM831X_TCH_Z_ENA, 0); |
| 188 | WM831X_TCH_Z_ENA, 0); | ||
| 189 | 216 | ||
| 190 | if (wm831x_ts->pen_down) | 217 | /* Make sure any pending IRQs are done, the above will prevent |
| 218 | * new ones firing. | ||
| 219 | */ | ||
| 220 | synchronize_irq(wm831x_ts->data_irq); | ||
| 221 | synchronize_irq(wm831x_ts->pd_irq); | ||
| 222 | |||
| 223 | /* Make sure the IRQ completion work is quiesced */ | ||
| 224 | flush_work_sync(&wm831x_ts->pd_data_work); | ||
| 225 | |||
| 226 | /* If we ended up with the pen down then make sure we revert back | ||
| 227 | * to pen detection state for the next time we start up. | ||
| 228 | */ | ||
| 229 | if (wm831x_ts->pen_down) { | ||
| 191 | disable_irq(wm831x_ts->data_irq); | 230 | disable_irq(wm831x_ts->data_irq); |
| 231 | enable_irq(wm831x_ts->pd_irq); | ||
| 232 | wm831x_ts->pen_down = false; | ||
| 233 | } | ||
| 192 | } | 234 | } |
| 193 | 235 | ||
| 194 | static __devinit int wm831x_ts_probe(struct platform_device *pdev) | 236 | static __devinit int wm831x_ts_probe(struct platform_device *pdev) |
| @@ -198,7 +240,7 @@ static __devinit int wm831x_ts_probe(struct platform_device *pdev) | |||
| 198 | struct wm831x_pdata *core_pdata = dev_get_platdata(pdev->dev.parent); | 240 | struct wm831x_pdata *core_pdata = dev_get_platdata(pdev->dev.parent); |
| 199 | struct wm831x_touch_pdata *pdata = NULL; | 241 | struct wm831x_touch_pdata *pdata = NULL; |
| 200 | struct input_dev *input_dev; | 242 | struct input_dev *input_dev; |
| 201 | int error; | 243 | int error, irqf; |
| 202 | 244 | ||
| 203 | if (core_pdata) | 245 | if (core_pdata) |
| 204 | pdata = core_pdata->touch; | 246 | pdata = core_pdata->touch; |
| @@ -212,6 +254,7 @@ static __devinit int wm831x_ts_probe(struct platform_device *pdev) | |||
| 212 | 254 | ||
| 213 | wm831x_ts->wm831x = wm831x; | 255 | wm831x_ts->wm831x = wm831x; |
| 214 | wm831x_ts->input_dev = input_dev; | 256 | wm831x_ts->input_dev = input_dev; |
| 257 | INIT_WORK(&wm831x_ts->pd_data_work, wm831x_pd_data_work); | ||
| 215 | 258 | ||
| 216 | /* | 259 | /* |
| 217 | * If we have a direct IRQ use it, otherwise use the interrupt | 260 | * If we have a direct IRQ use it, otherwise use the interrupt |
| @@ -270,9 +313,14 @@ static __devinit int wm831x_ts_probe(struct platform_device *pdev) | |||
| 270 | wm831x_set_bits(wm831x, WM831X_TOUCH_CONTROL_1, | 313 | wm831x_set_bits(wm831x, WM831X_TOUCH_CONTROL_1, |
| 271 | WM831X_TCH_RATE_MASK, 6); | 314 | WM831X_TCH_RATE_MASK, 6); |
| 272 | 315 | ||
| 316 | if (pdata && pdata->data_irqf) | ||
| 317 | irqf = pdata->data_irqf; | ||
| 318 | else | ||
| 319 | irqf = IRQF_TRIGGER_HIGH; | ||
| 320 | |||
| 273 | error = request_threaded_irq(wm831x_ts->data_irq, | 321 | error = request_threaded_irq(wm831x_ts->data_irq, |
| 274 | NULL, wm831x_ts_data_irq, | 322 | NULL, wm831x_ts_data_irq, |
| 275 | IRQF_ONESHOT, | 323 | irqf | IRQF_ONESHOT, |
| 276 | "Touchscreen data", wm831x_ts); | 324 | "Touchscreen data", wm831x_ts); |
| 277 | if (error) { | 325 | if (error) { |
| 278 | dev_err(&pdev->dev, "Failed to request data IRQ %d: %d\n", | 326 | dev_err(&pdev->dev, "Failed to request data IRQ %d: %d\n", |
| @@ -281,9 +329,14 @@ static __devinit int wm831x_ts_probe(struct platform_device *pdev) | |||
| 281 | } | 329 | } |
| 282 | disable_irq(wm831x_ts->data_irq); | 330 | disable_irq(wm831x_ts->data_irq); |
| 283 | 331 | ||
| 332 | if (pdata && pdata->pd_irqf) | ||
| 333 | irqf = pdata->pd_irqf; | ||
| 334 | else | ||
| 335 | irqf = IRQF_TRIGGER_HIGH; | ||
| 336 | |||
| 284 | error = request_threaded_irq(wm831x_ts->pd_irq, | 337 | error = request_threaded_irq(wm831x_ts->pd_irq, |
| 285 | NULL, wm831x_ts_pen_down_irq, | 338 | NULL, wm831x_ts_pen_down_irq, |
| 286 | IRQF_ONESHOT, | 339 | irqf | IRQF_ONESHOT, |
| 287 | "Touchscreen pen down", wm831x_ts); | 340 | "Touchscreen pen down", wm831x_ts); |
| 288 | if (error) { | 341 | if (error) { |
| 289 | dev_err(&pdev->dev, "Failed to request pen down IRQ %d: %d\n", | 342 | dev_err(&pdev->dev, "Failed to request pen down IRQ %d: %d\n", |
diff --git a/drivers/media/common/tuners/tda18271-common.c b/drivers/media/common/tuners/tda18271-common.c index 5466d47db899..aae40e52af5b 100644 --- a/drivers/media/common/tuners/tda18271-common.c +++ b/drivers/media/common/tuners/tda18271-common.c | |||
| @@ -533,16 +533,7 @@ int tda18271_calc_main_pll(struct dvb_frontend *fe, u32 freq) | |||
| 533 | if (tda_fail(ret)) | 533 | if (tda_fail(ret)) |
| 534 | goto fail; | 534 | goto fail; |
| 535 | 535 | ||
| 536 | regs[R_MPD] = (0x77 & pd); | 536 | regs[R_MPD] = (0x7f & pd); |
| 537 | |||
| 538 | switch (priv->mode) { | ||
| 539 | case TDA18271_ANALOG: | ||
| 540 | regs[R_MPD] &= ~0x08; | ||
| 541 | break; | ||
| 542 | case TDA18271_DIGITAL: | ||
| 543 | regs[R_MPD] |= 0x08; | ||
| 544 | break; | ||
| 545 | } | ||
| 546 | 537 | ||
| 547 | div = ((d * (freq / 1000)) << 7) / 125; | 538 | div = ((d * (freq / 1000)) << 7) / 125; |
| 548 | 539 | ||
diff --git a/drivers/media/common/tuners/tda18271-fe.c b/drivers/media/common/tuners/tda18271-fe.c index 9ad4454a148d..d884f5eee73c 100644 --- a/drivers/media/common/tuners/tda18271-fe.c +++ b/drivers/media/common/tuners/tda18271-fe.c | |||
| @@ -579,8 +579,8 @@ static int tda18271_rf_tracking_filters_init(struct dvb_frontend *fe, u32 freq) | |||
| 579 | #define RF3 2 | 579 | #define RF3 2 |
| 580 | u32 rf_default[3]; | 580 | u32 rf_default[3]; |
| 581 | u32 rf_freq[3]; | 581 | u32 rf_freq[3]; |
| 582 | u8 prog_cal[3]; | 582 | s32 prog_cal[3]; |
| 583 | u8 prog_tab[3]; | 583 | s32 prog_tab[3]; |
| 584 | 584 | ||
| 585 | i = tda18271_lookup_rf_band(fe, &freq, NULL); | 585 | i = tda18271_lookup_rf_band(fe, &freq, NULL); |
| 586 | 586 | ||
| @@ -602,32 +602,33 @@ static int tda18271_rf_tracking_filters_init(struct dvb_frontend *fe, u32 freq) | |||
| 602 | return bcal; | 602 | return bcal; |
| 603 | 603 | ||
| 604 | tda18271_calc_rf_cal(fe, &rf_freq[rf]); | 604 | tda18271_calc_rf_cal(fe, &rf_freq[rf]); |
| 605 | prog_tab[rf] = regs[R_EB14]; | 605 | prog_tab[rf] = (s32)regs[R_EB14]; |
| 606 | 606 | ||
| 607 | if (1 == bcal) | 607 | if (1 == bcal) |
| 608 | prog_cal[rf] = tda18271_calibrate_rf(fe, rf_freq[rf]); | 608 | prog_cal[rf] = |
| 609 | (s32)tda18271_calibrate_rf(fe, rf_freq[rf]); | ||
| 609 | else | 610 | else |
| 610 | prog_cal[rf] = prog_tab[rf]; | 611 | prog_cal[rf] = prog_tab[rf]; |
| 611 | 612 | ||
| 612 | switch (rf) { | 613 | switch (rf) { |
| 613 | case RF1: | 614 | case RF1: |
| 614 | map[i].rf_a1 = 0; | 615 | map[i].rf_a1 = 0; |
| 615 | map[i].rf_b1 = (s32)(prog_cal[RF1] - prog_tab[RF1]); | 616 | map[i].rf_b1 = (prog_cal[RF1] - prog_tab[RF1]); |
| 616 | map[i].rf1 = rf_freq[RF1] / 1000; | 617 | map[i].rf1 = rf_freq[RF1] / 1000; |
| 617 | break; | 618 | break; |
| 618 | case RF2: | 619 | case RF2: |
| 619 | dividend = (s32)(prog_cal[RF2] - prog_tab[RF2]) - | 620 | dividend = (prog_cal[RF2] - prog_tab[RF2] - |
| 620 | (s32)(prog_cal[RF1] + prog_tab[RF1]); | 621 | prog_cal[RF1] + prog_tab[RF1]); |
| 621 | divisor = (s32)(rf_freq[RF2] - rf_freq[RF1]) / 1000; | 622 | divisor = (s32)(rf_freq[RF2] - rf_freq[RF1]) / 1000; |
| 622 | map[i].rf_a1 = (dividend / divisor); | 623 | map[i].rf_a1 = (dividend / divisor); |
| 623 | map[i].rf2 = rf_freq[RF2] / 1000; | 624 | map[i].rf2 = rf_freq[RF2] / 1000; |
| 624 | break; | 625 | break; |
| 625 | case RF3: | 626 | case RF3: |
| 626 | dividend = (s32)(prog_cal[RF3] - prog_tab[RF3]) - | 627 | dividend = (prog_cal[RF3] - prog_tab[RF3] - |
| 627 | (s32)(prog_cal[RF2] + prog_tab[RF2]); | 628 | prog_cal[RF2] + prog_tab[RF2]); |
| 628 | divisor = (s32)(rf_freq[RF3] - rf_freq[RF2]) / 1000; | 629 | divisor = (s32)(rf_freq[RF3] - rf_freq[RF2]) / 1000; |
| 629 | map[i].rf_a2 = (dividend / divisor); | 630 | map[i].rf_a2 = (dividend / divisor); |
| 630 | map[i].rf_b2 = (s32)(prog_cal[RF2] - prog_tab[RF2]); | 631 | map[i].rf_b2 = (prog_cal[RF2] - prog_tab[RF2]); |
| 631 | map[i].rf3 = rf_freq[RF3] / 1000; | 632 | map[i].rf3 = rf_freq[RF3] / 1000; |
| 632 | break; | 633 | break; |
| 633 | default: | 634 | default: |
diff --git a/drivers/media/common/tuners/tda18271-maps.c b/drivers/media/common/tuners/tda18271-maps.c index e7f84c705da8..3d5b6ab7e332 100644 --- a/drivers/media/common/tuners/tda18271-maps.c +++ b/drivers/media/common/tuners/tda18271-maps.c | |||
| @@ -229,8 +229,7 @@ static struct tda18271_map tda18271c2_km[] = { | |||
| 229 | static struct tda18271_map tda18271_rf_band[] = { | 229 | static struct tda18271_map tda18271_rf_band[] = { |
| 230 | { .rfmax = 47900, .val = 0x00 }, | 230 | { .rfmax = 47900, .val = 0x00 }, |
| 231 | { .rfmax = 61100, .val = 0x01 }, | 231 | { .rfmax = 61100, .val = 0x01 }, |
| 232 | /* { .rfmax = 152600, .val = 0x02 }, */ | 232 | { .rfmax = 152600, .val = 0x02 }, |
| 233 | { .rfmax = 121200, .val = 0x02 }, | ||
| 234 | { .rfmax = 164700, .val = 0x03 }, | 233 | { .rfmax = 164700, .val = 0x03 }, |
| 235 | { .rfmax = 203500, .val = 0x04 }, | 234 | { .rfmax = 203500, .val = 0x04 }, |
| 236 | { .rfmax = 457800, .val = 0x05 }, | 235 | { .rfmax = 457800, .val = 0x05 }, |
| @@ -448,7 +447,7 @@ static struct tda18271_map tda18271c2_rf_cal[] = { | |||
| 448 | { .rfmax = 150000, .val = 0xb0 }, | 447 | { .rfmax = 150000, .val = 0xb0 }, |
| 449 | { .rfmax = 151000, .val = 0xb1 }, | 448 | { .rfmax = 151000, .val = 0xb1 }, |
| 450 | { .rfmax = 152000, .val = 0xb7 }, | 449 | { .rfmax = 152000, .val = 0xb7 }, |
| 451 | { .rfmax = 153000, .val = 0xbd }, | 450 | { .rfmax = 152600, .val = 0xbd }, |
| 452 | { .rfmax = 154000, .val = 0x20 }, | 451 | { .rfmax = 154000, .val = 0x20 }, |
| 453 | { .rfmax = 155000, .val = 0x22 }, | 452 | { .rfmax = 155000, .val = 0x22 }, |
| 454 | { .rfmax = 156000, .val = 0x24 }, | 453 | { .rfmax = 156000, .val = 0x24 }, |
| @@ -459,7 +458,7 @@ static struct tda18271_map tda18271c2_rf_cal[] = { | |||
| 459 | { .rfmax = 161000, .val = 0x2d }, | 458 | { .rfmax = 161000, .val = 0x2d }, |
| 460 | { .rfmax = 163000, .val = 0x2e }, | 459 | { .rfmax = 163000, .val = 0x2e }, |
| 461 | { .rfmax = 164000, .val = 0x2f }, | 460 | { .rfmax = 164000, .val = 0x2f }, |
| 462 | { .rfmax = 165000, .val = 0x30 }, | 461 | { .rfmax = 164700, .val = 0x30 }, |
| 463 | { .rfmax = 166000, .val = 0x11 }, | 462 | { .rfmax = 166000, .val = 0x11 }, |
| 464 | { .rfmax = 167000, .val = 0x12 }, | 463 | { .rfmax = 167000, .val = 0x12 }, |
| 465 | { .rfmax = 168000, .val = 0x13 }, | 464 | { .rfmax = 168000, .val = 0x13 }, |
| @@ -510,7 +509,8 @@ static struct tda18271_map tda18271c2_rf_cal[] = { | |||
| 510 | { .rfmax = 236000, .val = 0x1b }, | 509 | { .rfmax = 236000, .val = 0x1b }, |
| 511 | { .rfmax = 237000, .val = 0x1c }, | 510 | { .rfmax = 237000, .val = 0x1c }, |
| 512 | { .rfmax = 240000, .val = 0x1d }, | 511 | { .rfmax = 240000, .val = 0x1d }, |
| 513 | { .rfmax = 242000, .val = 0x1f }, | 512 | { .rfmax = 242000, .val = 0x1e }, |
| 513 | { .rfmax = 244000, .val = 0x1f }, | ||
| 514 | { .rfmax = 247000, .val = 0x20 }, | 514 | { .rfmax = 247000, .val = 0x20 }, |
| 515 | { .rfmax = 249000, .val = 0x21 }, | 515 | { .rfmax = 249000, .val = 0x21 }, |
| 516 | { .rfmax = 252000, .val = 0x22 }, | 516 | { .rfmax = 252000, .val = 0x22 }, |
| @@ -624,7 +624,7 @@ static struct tda18271_map tda18271c2_rf_cal[] = { | |||
| 624 | { .rfmax = 453000, .val = 0x93 }, | 624 | { .rfmax = 453000, .val = 0x93 }, |
| 625 | { .rfmax = 454000, .val = 0x94 }, | 625 | { .rfmax = 454000, .val = 0x94 }, |
| 626 | { .rfmax = 456000, .val = 0x96 }, | 626 | { .rfmax = 456000, .val = 0x96 }, |
| 627 | { .rfmax = 457000, .val = 0x98 }, | 627 | { .rfmax = 457800, .val = 0x98 }, |
| 628 | { .rfmax = 461000, .val = 0x11 }, | 628 | { .rfmax = 461000, .val = 0x11 }, |
| 629 | { .rfmax = 468000, .val = 0x12 }, | 629 | { .rfmax = 468000, .val = 0x12 }, |
| 630 | { .rfmax = 472000, .val = 0x13 }, | 630 | { .rfmax = 472000, .val = 0x13 }, |
diff --git a/drivers/media/dvb/b2c2/flexcop-pci.c b/drivers/media/dvb/b2c2/flexcop-pci.c index 955254090a0e..03f96d6ca894 100644 --- a/drivers/media/dvb/b2c2/flexcop-pci.c +++ b/drivers/media/dvb/b2c2/flexcop-pci.c | |||
| @@ -38,7 +38,7 @@ MODULE_PARM_DESC(debug, | |||
| 38 | DEBSTATUS); | 38 | DEBSTATUS); |
| 39 | 39 | ||
| 40 | #define DRIVER_VERSION "0.1" | 40 | #define DRIVER_VERSION "0.1" |
| 41 | #define DRIVER_NAME "Technisat/B2C2 FlexCop II/IIb/III Digital TV PCI Driver" | 41 | #define DRIVER_NAME "flexcop-pci" |
| 42 | #define DRIVER_AUTHOR "Patrick Boettcher <patrick.boettcher@desy.de>" | 42 | #define DRIVER_AUTHOR "Patrick Boettcher <patrick.boettcher@desy.de>" |
| 43 | 43 | ||
| 44 | struct flexcop_pci { | 44 | struct flexcop_pci { |
diff --git a/drivers/media/dvb/dvb-usb/Kconfig b/drivers/media/dvb/dvb-usb/Kconfig index fe4f894183ff..ccbd39a38c46 100644 --- a/drivers/media/dvb/dvb-usb/Kconfig +++ b/drivers/media/dvb/dvb-usb/Kconfig | |||
| @@ -362,7 +362,7 @@ config DVB_USB_LME2510 | |||
| 362 | config DVB_USB_TECHNISAT_USB2 | 362 | config DVB_USB_TECHNISAT_USB2 |
| 363 | tristate "Technisat DVB-S/S2 USB2.0 support" | 363 | tristate "Technisat DVB-S/S2 USB2.0 support" |
| 364 | depends on DVB_USB | 364 | depends on DVB_USB |
| 365 | select DVB_STB0899 if !DVB_FE_CUSTOMISE | 365 | select DVB_STV090x if !DVB_FE_CUSTOMISE |
| 366 | select DVB_STB6100 if !DVB_FE_CUSTOMISE | 366 | select DVB_STV6110x if !DVB_FE_CUSTOMISE |
| 367 | help | 367 | help |
| 368 | Say Y here to support the Technisat USB2 DVB-S/S2 device | 368 | Say Y here to support the Technisat USB2 DVB-S/S2 device |
diff --git a/drivers/media/dvb/dvb-usb/dib0700_devices.c b/drivers/media/dvb/dvb-usb/dib0700_devices.c index 97af266d7f1d..65214af5cd74 100644 --- a/drivers/media/dvb/dvb-usb/dib0700_devices.c +++ b/drivers/media/dvb/dvb-usb/dib0700_devices.c | |||
| @@ -2162,7 +2162,7 @@ struct dibx000_agc_config dib7090_agc_config[2] = { | |||
| 2162 | .agc1_pt3 = 98, | 2162 | .agc1_pt3 = 98, |
| 2163 | .agc1_slope1 = 0, | 2163 | .agc1_slope1 = 0, |
| 2164 | .agc1_slope2 = 167, | 2164 | .agc1_slope2 = 167, |
| 2165 | .agc1_pt1 = 98, | 2165 | .agc2_pt1 = 98, |
| 2166 | .agc2_pt2 = 255, | 2166 | .agc2_pt2 = 255, |
| 2167 | .agc2_slope1 = 104, | 2167 | .agc2_slope1 = 104, |
| 2168 | .agc2_slope2 = 0, | 2168 | .agc2_slope2 = 0, |
| @@ -2440,11 +2440,11 @@ static int tfe7090pvr_frontend0_attach(struct dvb_usb_adapter *adap) | |||
| 2440 | dib0700_set_i2c_speed(adap->dev, 340); | 2440 | dib0700_set_i2c_speed(adap->dev, 340); |
| 2441 | adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 0x90, &tfe7090pvr_dib7000p_config[0]); | 2441 | adap->fe = dvb_attach(dib7000p_attach, &adap->dev->i2c_adap, 0x90, &tfe7090pvr_dib7000p_config[0]); |
| 2442 | 2442 | ||
| 2443 | dib7090_slave_reset(adap->fe); | ||
| 2444 | |||
| 2445 | if (adap->fe == NULL) | 2443 | if (adap->fe == NULL) |
| 2446 | return -ENODEV; | 2444 | return -ENODEV; |
| 2447 | 2445 | ||
| 2446 | dib7090_slave_reset(adap->fe); | ||
| 2447 | |||
| 2448 | return 0; | 2448 | return 0; |
| 2449 | } | 2449 | } |
| 2450 | 2450 | ||
diff --git a/drivers/media/media-entity.c b/drivers/media/media-entity.c index 23640ed44d85..056138f63c7d 100644 --- a/drivers/media/media-entity.c +++ b/drivers/media/media-entity.c | |||
| @@ -378,7 +378,6 @@ EXPORT_SYMBOL_GPL(media_entity_create_link); | |||
| 378 | 378 | ||
| 379 | static int __media_entity_setup_link_notify(struct media_link *link, u32 flags) | 379 | static int __media_entity_setup_link_notify(struct media_link *link, u32 flags) |
| 380 | { | 380 | { |
| 381 | const u32 mask = MEDIA_LNK_FL_ENABLED; | ||
| 382 | int ret; | 381 | int ret; |
| 383 | 382 | ||
| 384 | /* Notify both entities. */ | 383 | /* Notify both entities. */ |
| @@ -395,7 +394,7 @@ static int __media_entity_setup_link_notify(struct media_link *link, u32 flags) | |||
| 395 | return ret; | 394 | return ret; |
| 396 | } | 395 | } |
| 397 | 396 | ||
| 398 | link->flags = (link->flags & ~mask) | (flags & mask); | 397 | link->flags = flags; |
| 399 | link->reverse->flags = link->flags; | 398 | link->reverse->flags = link->flags; |
| 400 | 399 | ||
| 401 | return 0; | 400 | return 0; |
| @@ -417,6 +416,7 @@ static int __media_entity_setup_link_notify(struct media_link *link, u32 flags) | |||
| 417 | */ | 416 | */ |
| 418 | int __media_entity_setup_link(struct media_link *link, u32 flags) | 417 | int __media_entity_setup_link(struct media_link *link, u32 flags) |
| 419 | { | 418 | { |
| 419 | const u32 mask = MEDIA_LNK_FL_ENABLED; | ||
| 420 | struct media_device *mdev; | 420 | struct media_device *mdev; |
| 421 | struct media_entity *source, *sink; | 421 | struct media_entity *source, *sink; |
| 422 | int ret = -EBUSY; | 422 | int ret = -EBUSY; |
| @@ -424,6 +424,10 @@ int __media_entity_setup_link(struct media_link *link, u32 flags) | |||
| 424 | if (link == NULL) | 424 | if (link == NULL) |
| 425 | return -EINVAL; | 425 | return -EINVAL; |
| 426 | 426 | ||
| 427 | /* The non-modifiable link flags must not be modified. */ | ||
| 428 | if ((link->flags & ~mask) != (flags & ~mask)) | ||
| 429 | return -EINVAL; | ||
| 430 | |||
| 427 | if (link->flags & MEDIA_LNK_FL_IMMUTABLE) | 431 | if (link->flags & MEDIA_LNK_FL_IMMUTABLE) |
| 428 | return link->flags == flags ? 0 : -EINVAL; | 432 | return link->flags == flags ? 0 : -EINVAL; |
| 429 | 433 | ||
diff --git a/drivers/media/radio/radio-sf16fmr2.c b/drivers/media/radio/radio-sf16fmr2.c index dc3f04c52d5e..87bad7678d92 100644 --- a/drivers/media/radio/radio-sf16fmr2.c +++ b/drivers/media/radio/radio-sf16fmr2.c | |||
| @@ -170,7 +170,7 @@ static int fmr2_setfreq(struct fmr2 *dev) | |||
| 170 | return 0; | 170 | return 0; |
| 171 | } | 171 | } |
| 172 | 172 | ||
| 173 | /* !!! not tested, in my card this does't work !!! */ | 173 | /* !!! not tested, in my card this doesn't work !!! */ |
| 174 | static int fmr2_setvolume(struct fmr2 *dev) | 174 | static int fmr2_setvolume(struct fmr2 *dev) |
| 175 | { | 175 | { |
| 176 | int vol[16] = { 0x021, 0x084, 0x090, 0x104, | 176 | int vol[16] = { 0x021, 0x084, 0x090, 0x104, |
diff --git a/drivers/media/video/Kconfig b/drivers/media/video/Kconfig index 4498b944dec8..00f51dd121f3 100644 --- a/drivers/media/video/Kconfig +++ b/drivers/media/video/Kconfig | |||
| @@ -875,7 +875,7 @@ config MX3_VIDEO | |||
| 875 | config VIDEO_MX3 | 875 | config VIDEO_MX3 |
| 876 | tristate "i.MX3x Camera Sensor Interface driver" | 876 | tristate "i.MX3x Camera Sensor Interface driver" |
| 877 | depends on VIDEO_DEV && MX3_IPU && SOC_CAMERA | 877 | depends on VIDEO_DEV && MX3_IPU && SOC_CAMERA |
| 878 | select VIDEOBUF_DMA_CONTIG | 878 | select VIDEOBUF2_DMA_CONTIG |
| 879 | select MX3_VIDEO | 879 | select MX3_VIDEO |
| 880 | ---help--- | 880 | ---help--- |
| 881 | This is a v4l2 driver for the i.MX3x Camera Sensor Interface | 881 | This is a v4l2 driver for the i.MX3x Camera Sensor Interface |
diff --git a/drivers/media/video/cx18/cx18-streams.c b/drivers/media/video/cx18/cx18-streams.c index c6e2ca3b1149..6fbc356113c1 100644 --- a/drivers/media/video/cx18/cx18-streams.c +++ b/drivers/media/video/cx18/cx18-streams.c | |||
| @@ -350,9 +350,17 @@ void cx18_streams_cleanup(struct cx18 *cx, int unregister) | |||
| 350 | 350 | ||
| 351 | /* No struct video_device, but can have buffers allocated */ | 351 | /* No struct video_device, but can have buffers allocated */ |
| 352 | if (type == CX18_ENC_STREAM_TYPE_IDX) { | 352 | if (type == CX18_ENC_STREAM_TYPE_IDX) { |
| 353 | /* If the module params didn't inhibit IDX ... */ | ||
| 353 | if (cx->stream_buffers[type] != 0) { | 354 | if (cx->stream_buffers[type] != 0) { |
| 354 | cx->stream_buffers[type] = 0; | 355 | cx->stream_buffers[type] = 0; |
| 355 | cx18_stream_free(&cx->streams[type]); | 356 | /* |
| 357 | * Before calling cx18_stream_free(), | ||
| 358 | * check if the IDX stream was actually set up. | ||
| 359 | * Needed, since the cx18_probe() error path | ||
| 360 | * exits through here as well as normal clean up | ||
| 361 | */ | ||
| 362 | if (cx->streams[type].buffers != 0) | ||
| 363 | cx18_stream_free(&cx->streams[type]); | ||
| 356 | } | 364 | } |
| 357 | continue; | 365 | continue; |
| 358 | } | 366 | } |
diff --git a/drivers/media/video/cx23885/Kconfig b/drivers/media/video/cx23885/Kconfig index 3b6e7f28568e..caab1bfb79e2 100644 --- a/drivers/media/video/cx23885/Kconfig +++ b/drivers/media/video/cx23885/Kconfig | |||
| @@ -22,6 +22,7 @@ config VIDEO_CX23885 | |||
| 22 | select DVB_CX24116 if !DVB_FE_CUSTOMISE | 22 | select DVB_CX24116 if !DVB_FE_CUSTOMISE |
| 23 | select DVB_STV0900 if !DVB_FE_CUSTOMISE | 23 | select DVB_STV0900 if !DVB_FE_CUSTOMISE |
| 24 | select DVB_DS3000 if !DVB_FE_CUSTOMISE | 24 | select DVB_DS3000 if !DVB_FE_CUSTOMISE |
| 25 | select DVB_STV0367 if !DVB_FE_CUSTOMISE | ||
| 25 | select MEDIA_TUNER_MT2131 if !MEDIA_TUNER_CUSTOMISE | 26 | select MEDIA_TUNER_MT2131 if !MEDIA_TUNER_CUSTOMISE |
| 26 | select MEDIA_TUNER_XC2028 if !MEDIA_TUNER_CUSTOMISE | 27 | select MEDIA_TUNER_XC2028 if !MEDIA_TUNER_CUSTOMISE |
| 27 | select MEDIA_TUNER_TDA8290 if !MEDIA_TUNER_CUSTOMISE | 28 | select MEDIA_TUNER_TDA8290 if !MEDIA_TUNER_CUSTOMISE |
diff --git a/drivers/media/video/imx074.c b/drivers/media/video/imx074.c index 1a1169115716..0382ea752e6f 100644 --- a/drivers/media/video/imx074.c +++ b/drivers/media/video/imx074.c | |||
| @@ -298,7 +298,7 @@ static unsigned long imx074_query_bus_param(struct soc_camera_device *icd) | |||
| 298 | static int imx074_set_bus_param(struct soc_camera_device *icd, | 298 | static int imx074_set_bus_param(struct soc_camera_device *icd, |
| 299 | unsigned long flags) | 299 | unsigned long flags) |
| 300 | { | 300 | { |
| 301 | return -1; | 301 | return -EINVAL; |
| 302 | } | 302 | } |
| 303 | 303 | ||
| 304 | static struct soc_camera_ops imx074_ops = { | 304 | static struct soc_camera_ops imx074_ops = { |
diff --git a/drivers/media/video/omap3isp/isp.c b/drivers/media/video/omap3isp/isp.c index 503bd7922bd6..472a69359e60 100644 --- a/drivers/media/video/omap3isp/isp.c +++ b/drivers/media/video/omap3isp/isp.c | |||
| @@ -215,20 +215,21 @@ static u32 isp_set_xclk(struct isp_device *isp, u32 xclk, u8 xclksel) | |||
| 215 | } | 215 | } |
| 216 | 216 | ||
| 217 | switch (xclksel) { | 217 | switch (xclksel) { |
| 218 | case 0: | 218 | case ISP_XCLK_A: |
| 219 | isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL, | 219 | isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL, |
| 220 | ISPTCTRL_CTRL_DIVA_MASK, | 220 | ISPTCTRL_CTRL_DIVA_MASK, |
| 221 | divisor << ISPTCTRL_CTRL_DIVA_SHIFT); | 221 | divisor << ISPTCTRL_CTRL_DIVA_SHIFT); |
| 222 | dev_dbg(isp->dev, "isp_set_xclk(): cam_xclka set to %d Hz\n", | 222 | dev_dbg(isp->dev, "isp_set_xclk(): cam_xclka set to %d Hz\n", |
| 223 | currentxclk); | 223 | currentxclk); |
| 224 | break; | 224 | break; |
| 225 | case 1: | 225 | case ISP_XCLK_B: |
| 226 | isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL, | 226 | isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL, |
| 227 | ISPTCTRL_CTRL_DIVB_MASK, | 227 | ISPTCTRL_CTRL_DIVB_MASK, |
| 228 | divisor << ISPTCTRL_CTRL_DIVB_SHIFT); | 228 | divisor << ISPTCTRL_CTRL_DIVB_SHIFT); |
| 229 | dev_dbg(isp->dev, "isp_set_xclk(): cam_xclkb set to %d Hz\n", | 229 | dev_dbg(isp->dev, "isp_set_xclk(): cam_xclkb set to %d Hz\n", |
| 230 | currentxclk); | 230 | currentxclk); |
| 231 | break; | 231 | break; |
| 232 | case ISP_XCLK_NONE: | ||
| 232 | default: | 233 | default: |
| 233 | omap3isp_put(isp); | 234 | omap3isp_put(isp); |
| 234 | dev_dbg(isp->dev, "ISP_ERR: isp_set_xclk(): Invalid requested " | 235 | dev_dbg(isp->dev, "ISP_ERR: isp_set_xclk(): Invalid requested " |
| @@ -237,13 +238,13 @@ static u32 isp_set_xclk(struct isp_device *isp, u32 xclk, u8 xclksel) | |||
| 237 | } | 238 | } |
| 238 | 239 | ||
| 239 | /* Do we go from stable whatever to clock? */ | 240 | /* Do we go from stable whatever to clock? */ |
| 240 | if (divisor >= 2 && isp->xclk_divisor[xclksel] < 2) | 241 | if (divisor >= 2 && isp->xclk_divisor[xclksel - 1] < 2) |
| 241 | omap3isp_get(isp); | 242 | omap3isp_get(isp); |
| 242 | /* Stopping the clock. */ | 243 | /* Stopping the clock. */ |
| 243 | else if (divisor < 2 && isp->xclk_divisor[xclksel] >= 2) | 244 | else if (divisor < 2 && isp->xclk_divisor[xclksel - 1] >= 2) |
| 244 | omap3isp_put(isp); | 245 | omap3isp_put(isp); |
| 245 | 246 | ||
| 246 | isp->xclk_divisor[xclksel] = divisor; | 247 | isp->xclk_divisor[xclksel - 1] = divisor; |
| 247 | 248 | ||
| 248 | omap3isp_put(isp); | 249 | omap3isp_put(isp); |
| 249 | 250 | ||
| @@ -285,7 +286,8 @@ static void isp_power_settings(struct isp_device *isp, int idle) | |||
| 285 | */ | 286 | */ |
| 286 | void omap3isp_configure_bridge(struct isp_device *isp, | 287 | void omap3isp_configure_bridge(struct isp_device *isp, |
| 287 | enum ccdc_input_entity input, | 288 | enum ccdc_input_entity input, |
| 288 | const struct isp_parallel_platform_data *pdata) | 289 | const struct isp_parallel_platform_data *pdata, |
| 290 | unsigned int shift) | ||
| 289 | { | 291 | { |
| 290 | u32 ispctrl_val; | 292 | u32 ispctrl_val; |
| 291 | 293 | ||
| @@ -298,9 +300,9 @@ void omap3isp_configure_bridge(struct isp_device *isp, | |||
| 298 | switch (input) { | 300 | switch (input) { |
| 299 | case CCDC_INPUT_PARALLEL: | 301 | case CCDC_INPUT_PARALLEL: |
| 300 | ispctrl_val |= ISPCTRL_PAR_SER_CLK_SEL_PARALLEL; | 302 | ispctrl_val |= ISPCTRL_PAR_SER_CLK_SEL_PARALLEL; |
| 301 | ispctrl_val |= pdata->data_lane_shift << ISPCTRL_SHIFT_SHIFT; | ||
| 302 | ispctrl_val |= pdata->clk_pol << ISPCTRL_PAR_CLK_POL_SHIFT; | 303 | ispctrl_val |= pdata->clk_pol << ISPCTRL_PAR_CLK_POL_SHIFT; |
| 303 | ispctrl_val |= pdata->bridge << ISPCTRL_PAR_BRIDGE_SHIFT; | 304 | ispctrl_val |= pdata->bridge << ISPCTRL_PAR_BRIDGE_SHIFT; |
| 305 | shift += pdata->data_lane_shift * 2; | ||
| 304 | break; | 306 | break; |
| 305 | 307 | ||
| 306 | case CCDC_INPUT_CSI2A: | 308 | case CCDC_INPUT_CSI2A: |
| @@ -319,6 +321,8 @@ void omap3isp_configure_bridge(struct isp_device *isp, | |||
| 319 | return; | 321 | return; |
| 320 | } | 322 | } |
| 321 | 323 | ||
| 324 | ispctrl_val |= ((shift/2) << ISPCTRL_SHIFT_SHIFT) & ISPCTRL_SHIFT_MASK; | ||
| 325 | |||
| 322 | ispctrl_val &= ~ISPCTRL_SYNC_DETECT_MASK; | 326 | ispctrl_val &= ~ISPCTRL_SYNC_DETECT_MASK; |
| 323 | ispctrl_val |= ISPCTRL_SYNC_DETECT_VSRISE; | 327 | ispctrl_val |= ISPCTRL_SYNC_DETECT_VSRISE; |
| 324 | 328 | ||
| @@ -658,6 +662,8 @@ int omap3isp_pipeline_pm_use(struct media_entity *entity, int use) | |||
| 658 | 662 | ||
| 659 | /* Apply power change to connected non-nodes. */ | 663 | /* Apply power change to connected non-nodes. */ |
| 660 | ret = isp_pipeline_pm_power(entity, change); | 664 | ret = isp_pipeline_pm_power(entity, change); |
| 665 | if (ret < 0) | ||
| 666 | entity->use_count -= change; | ||
| 661 | 667 | ||
| 662 | mutex_unlock(&entity->parent->graph_mutex); | 668 | mutex_unlock(&entity->parent->graph_mutex); |
| 663 | 669 | ||
| @@ -872,6 +878,9 @@ static int isp_pipeline_disable(struct isp_pipeline *pipe) | |||
| 872 | } | 878 | } |
| 873 | } | 879 | } |
| 874 | 880 | ||
| 881 | if (failure < 0) | ||
| 882 | isp->needs_reset = true; | ||
| 883 | |||
| 875 | return failure; | 884 | return failure; |
| 876 | } | 885 | } |
| 877 | 886 | ||
| @@ -884,7 +893,8 @@ static int isp_pipeline_disable(struct isp_pipeline *pipe) | |||
| 884 | * single-shot or continuous mode. | 893 | * single-shot or continuous mode. |
| 885 | * | 894 | * |
| 886 | * Return 0 if successful, or the return value of the failed video::s_stream | 895 | * Return 0 if successful, or the return value of the failed video::s_stream |
| 887 | * operation otherwise. | 896 | * operation otherwise. The pipeline state is not updated when the operation |
| 897 | * fails, except when stopping the pipeline. | ||
| 888 | */ | 898 | */ |
| 889 | int omap3isp_pipeline_set_stream(struct isp_pipeline *pipe, | 899 | int omap3isp_pipeline_set_stream(struct isp_pipeline *pipe, |
| 890 | enum isp_pipeline_stream_state state) | 900 | enum isp_pipeline_stream_state state) |
| @@ -895,7 +905,9 @@ int omap3isp_pipeline_set_stream(struct isp_pipeline *pipe, | |||
| 895 | ret = isp_pipeline_disable(pipe); | 905 | ret = isp_pipeline_disable(pipe); |
| 896 | else | 906 | else |
| 897 | ret = isp_pipeline_enable(pipe, state); | 907 | ret = isp_pipeline_enable(pipe, state); |
| 898 | pipe->stream_state = state; | 908 | |
| 909 | if (ret == 0 || state == ISP_PIPELINE_STREAM_STOPPED) | ||
| 910 | pipe->stream_state = state; | ||
| 899 | 911 | ||
| 900 | return ret; | 912 | return ret; |
| 901 | } | 913 | } |
| @@ -1481,6 +1493,10 @@ void omap3isp_put(struct isp_device *isp) | |||
| 1481 | if (--isp->ref_count == 0) { | 1493 | if (--isp->ref_count == 0) { |
| 1482 | isp_disable_interrupts(isp); | 1494 | isp_disable_interrupts(isp); |
| 1483 | isp_save_ctx(isp); | 1495 | isp_save_ctx(isp); |
| 1496 | if (isp->needs_reset) { | ||
| 1497 | isp_reset(isp); | ||
| 1498 | isp->needs_reset = false; | ||
| 1499 | } | ||
| 1484 | isp_disable_clocks(isp); | 1500 | isp_disable_clocks(isp); |
| 1485 | } | 1501 | } |
| 1486 | mutex_unlock(&isp->isp_mutex); | 1502 | mutex_unlock(&isp->isp_mutex); |
diff --git a/drivers/media/video/omap3isp/isp.h b/drivers/media/video/omap3isp/isp.h index cf5214e95a92..2620c405f5e4 100644 --- a/drivers/media/video/omap3isp/isp.h +++ b/drivers/media/video/omap3isp/isp.h | |||
| @@ -132,7 +132,6 @@ struct isp_reg { | |||
| 132 | 132 | ||
| 133 | /** | 133 | /** |
| 134 | * struct isp_parallel_platform_data - Parallel interface platform data | 134 | * struct isp_parallel_platform_data - Parallel interface platform data |
| 135 | * @width: Parallel bus width in bits (8, 10, 11 or 12) | ||
| 136 | * @data_lane_shift: Data lane shifter | 135 | * @data_lane_shift: Data lane shifter |
| 137 | * 0 - CAMEXT[13:0] -> CAM[13:0] | 136 | * 0 - CAMEXT[13:0] -> CAM[13:0] |
| 138 | * 1 - CAMEXT[13:2] -> CAM[11:0] | 137 | * 1 - CAMEXT[13:2] -> CAM[11:0] |
| @@ -146,7 +145,6 @@ struct isp_reg { | |||
| 146 | * ISPCTRL_PAR_BRIDGE_BENDIAN - Big endian | 145 | * ISPCTRL_PAR_BRIDGE_BENDIAN - Big endian |
| 147 | */ | 146 | */ |
| 148 | struct isp_parallel_platform_data { | 147 | struct isp_parallel_platform_data { |
| 149 | unsigned int width; | ||
| 150 | unsigned int data_lane_shift:2; | 148 | unsigned int data_lane_shift:2; |
| 151 | unsigned int clk_pol:1; | 149 | unsigned int clk_pol:1; |
| 152 | unsigned int bridge:4; | 150 | unsigned int bridge:4; |
| @@ -262,6 +260,7 @@ struct isp_device { | |||
| 262 | /* ISP Obj */ | 260 | /* ISP Obj */ |
| 263 | spinlock_t stat_lock; /* common lock for statistic drivers */ | 261 | spinlock_t stat_lock; /* common lock for statistic drivers */ |
| 264 | struct mutex isp_mutex; /* For handling ref_count field */ | 262 | struct mutex isp_mutex; /* For handling ref_count field */ |
| 263 | bool needs_reset; | ||
| 265 | int has_context; | 264 | int has_context; |
| 266 | int ref_count; | 265 | int ref_count; |
| 267 | unsigned int autoidle; | 266 | unsigned int autoidle; |
| @@ -311,11 +310,12 @@ int omap3isp_pipeline_set_stream(struct isp_pipeline *pipe, | |||
| 311 | enum isp_pipeline_stream_state state); | 310 | enum isp_pipeline_stream_state state); |
| 312 | void omap3isp_configure_bridge(struct isp_device *isp, | 311 | void omap3isp_configure_bridge(struct isp_device *isp, |
| 313 | enum ccdc_input_entity input, | 312 | enum ccdc_input_entity input, |
| 314 | const struct isp_parallel_platform_data *pdata); | 313 | const struct isp_parallel_platform_data *pdata, |
| 314 | unsigned int shift); | ||
| 315 | 315 | ||
| 316 | #define ISP_XCLK_NONE -1 | 316 | #define ISP_XCLK_NONE 0 |
| 317 | #define ISP_XCLK_A 0 | 317 | #define ISP_XCLK_A 1 |
| 318 | #define ISP_XCLK_B 1 | 318 | #define ISP_XCLK_B 2 |
| 319 | 319 | ||
| 320 | struct isp_device *omap3isp_get(struct isp_device *isp); | 320 | struct isp_device *omap3isp_get(struct isp_device *isp); |
| 321 | void omap3isp_put(struct isp_device *isp); | 321 | void omap3isp_put(struct isp_device *isp); |
diff --git a/drivers/media/video/omap3isp/ispccdc.c b/drivers/media/video/omap3isp/ispccdc.c index 5ff9d14ce710..39d501bda636 100644 --- a/drivers/media/video/omap3isp/ispccdc.c +++ b/drivers/media/video/omap3isp/ispccdc.c | |||
| @@ -43,6 +43,12 @@ __ccdc_get_format(struct isp_ccdc_device *ccdc, struct v4l2_subdev_fh *fh, | |||
| 43 | 43 | ||
| 44 | static const unsigned int ccdc_fmts[] = { | 44 | static const unsigned int ccdc_fmts[] = { |
| 45 | V4L2_MBUS_FMT_Y8_1X8, | 45 | V4L2_MBUS_FMT_Y8_1X8, |
| 46 | V4L2_MBUS_FMT_Y10_1X10, | ||
| 47 | V4L2_MBUS_FMT_Y12_1X12, | ||
| 48 | V4L2_MBUS_FMT_SGRBG8_1X8, | ||
| 49 | V4L2_MBUS_FMT_SRGGB8_1X8, | ||
| 50 | V4L2_MBUS_FMT_SBGGR8_1X8, | ||
| 51 | V4L2_MBUS_FMT_SGBRG8_1X8, | ||
| 46 | V4L2_MBUS_FMT_SGRBG10_1X10, | 52 | V4L2_MBUS_FMT_SGRBG10_1X10, |
| 47 | V4L2_MBUS_FMT_SRGGB10_1X10, | 53 | V4L2_MBUS_FMT_SRGGB10_1X10, |
| 48 | V4L2_MBUS_FMT_SBGGR10_1X10, | 54 | V4L2_MBUS_FMT_SBGGR10_1X10, |
| @@ -1110,21 +1116,38 @@ static void ccdc_configure(struct isp_ccdc_device *ccdc) | |||
| 1110 | struct isp_parallel_platform_data *pdata = NULL; | 1116 | struct isp_parallel_platform_data *pdata = NULL; |
| 1111 | struct v4l2_subdev *sensor; | 1117 | struct v4l2_subdev *sensor; |
| 1112 | struct v4l2_mbus_framefmt *format; | 1118 | struct v4l2_mbus_framefmt *format; |
| 1119 | const struct isp_format_info *fmt_info; | ||
| 1120 | struct v4l2_subdev_format fmt_src; | ||
| 1121 | unsigned int depth_out; | ||
| 1122 | unsigned int depth_in = 0; | ||
| 1113 | struct media_pad *pad; | 1123 | struct media_pad *pad; |
| 1114 | unsigned long flags; | 1124 | unsigned long flags; |
| 1125 | unsigned int shift; | ||
| 1115 | u32 syn_mode; | 1126 | u32 syn_mode; |
| 1116 | u32 ccdc_pattern; | 1127 | u32 ccdc_pattern; |
| 1117 | 1128 | ||
| 1118 | if (ccdc->input == CCDC_INPUT_PARALLEL) { | 1129 | pad = media_entity_remote_source(&ccdc->pads[CCDC_PAD_SINK]); |
| 1119 | pad = media_entity_remote_source(&ccdc->pads[CCDC_PAD_SINK]); | 1130 | sensor = media_entity_to_v4l2_subdev(pad->entity); |
| 1120 | sensor = media_entity_to_v4l2_subdev(pad->entity); | 1131 | if (ccdc->input == CCDC_INPUT_PARALLEL) |
| 1121 | pdata = &((struct isp_v4l2_subdevs_group *)sensor->host_priv) | 1132 | pdata = &((struct isp_v4l2_subdevs_group *)sensor->host_priv) |
| 1122 | ->bus.parallel; | 1133 | ->bus.parallel; |
| 1134 | |||
| 1135 | /* Compute shift value for lane shifter to configure the bridge. */ | ||
| 1136 | fmt_src.pad = pad->index; | ||
| 1137 | fmt_src.which = V4L2_SUBDEV_FORMAT_ACTIVE; | ||
| 1138 | if (!v4l2_subdev_call(sensor, pad, get_fmt, NULL, &fmt_src)) { | ||
| 1139 | fmt_info = omap3isp_video_format_info(fmt_src.format.code); | ||
| 1140 | depth_in = fmt_info->bpp; | ||
| 1123 | } | 1141 | } |
| 1124 | 1142 | ||
| 1125 | omap3isp_configure_bridge(isp, ccdc->input, pdata); | 1143 | fmt_info = omap3isp_video_format_info |
| 1144 | (isp->isp_ccdc.formats[CCDC_PAD_SINK].code); | ||
| 1145 | depth_out = fmt_info->bpp; | ||
| 1146 | |||
| 1147 | shift = depth_in - depth_out; | ||
| 1148 | omap3isp_configure_bridge(isp, ccdc->input, pdata, shift); | ||
| 1126 | 1149 | ||
| 1127 | ccdc->syncif.datsz = pdata ? pdata->width : 10; | 1150 | ccdc->syncif.datsz = depth_out; |
| 1128 | ccdc_config_sync_if(ccdc, &ccdc->syncif); | 1151 | ccdc_config_sync_if(ccdc, &ccdc->syncif); |
| 1129 | 1152 | ||
| 1130 | /* CCDC_PAD_SINK */ | 1153 | /* CCDC_PAD_SINK */ |
| @@ -1338,7 +1361,7 @@ static int ccdc_sbl_wait_idle(struct isp_ccdc_device *ccdc, | |||
| 1338 | * @ccdc: Pointer to ISP CCDC device. | 1361 | * @ccdc: Pointer to ISP CCDC device. |
| 1339 | * @event: Pointing which event trigger handler | 1362 | * @event: Pointing which event trigger handler |
| 1340 | * | 1363 | * |
| 1341 | * Return 1 when the event and stopping request combination is satisfyied, | 1364 | * Return 1 when the event and stopping request combination is satisfied, |
| 1342 | * zero otherwise. | 1365 | * zero otherwise. |
| 1343 | */ | 1366 | */ |
| 1344 | static int __ccdc_handle_stopping(struct isp_ccdc_device *ccdc, u32 event) | 1367 | static int __ccdc_handle_stopping(struct isp_ccdc_device *ccdc, u32 event) |
| @@ -1618,7 +1641,7 @@ static int ccdc_video_queue(struct isp_video *video, struct isp_buffer *buffer) | |||
| 1618 | 1641 | ||
| 1619 | ccdc_set_outaddr(ccdc, buffer->isp_addr); | 1642 | ccdc_set_outaddr(ccdc, buffer->isp_addr); |
| 1620 | 1643 | ||
| 1621 | /* We now have a buffer queued on the output, restart the pipeline in | 1644 | /* We now have a buffer queued on the output, restart the pipeline |
| 1622 | * on the next CCDC interrupt if running in continuous mode (or when | 1645 | * on the next CCDC interrupt if running in continuous mode (or when |
| 1623 | * starting the stream). | 1646 | * starting the stream). |
| 1624 | */ | 1647 | */ |
diff --git a/drivers/media/video/omap3isp/isppreview.c b/drivers/media/video/omap3isp/isppreview.c index 2b16988a501d..aba537af87e4 100644 --- a/drivers/media/video/omap3isp/isppreview.c +++ b/drivers/media/video/omap3isp/isppreview.c | |||
| @@ -755,7 +755,7 @@ static struct preview_update update_attrs[] = { | |||
| 755 | * @configs - pointer to update config structure. | 755 | * @configs - pointer to update config structure. |
| 756 | * @config - return pointer to appropriate structure field. | 756 | * @config - return pointer to appropriate structure field. |
| 757 | * @bit - for which feature to return pointers. | 757 | * @bit - for which feature to return pointers. |
| 758 | * Return size of coresponding prev_params member | 758 | * Return size of corresponding prev_params member |
| 759 | */ | 759 | */ |
| 760 | static u32 | 760 | static u32 |
| 761 | __preview_get_ptrs(struct prev_params *params, void **param, | 761 | __preview_get_ptrs(struct prev_params *params, void **param, |
diff --git a/drivers/media/video/omap3isp/ispqueue.c b/drivers/media/video/omap3isp/ispqueue.c index 8fddc5806b0d..9c317148205f 100644 --- a/drivers/media/video/omap3isp/ispqueue.c +++ b/drivers/media/video/omap3isp/ispqueue.c | |||
| @@ -339,7 +339,7 @@ static int isp_video_buffer_prepare_user(struct isp_video_buffer *buf) | |||
| 339 | up_read(¤t->mm->mmap_sem); | 339 | up_read(¤t->mm->mmap_sem); |
| 340 | 340 | ||
| 341 | if (ret != buf->npages) { | 341 | if (ret != buf->npages) { |
| 342 | buf->npages = ret; | 342 | buf->npages = ret < 0 ? 0 : ret; |
| 343 | isp_video_buffer_cleanup(buf); | 343 | isp_video_buffer_cleanup(buf); |
| 344 | return -EFAULT; | 344 | return -EFAULT; |
| 345 | } | 345 | } |
| @@ -408,8 +408,8 @@ done: | |||
| 408 | * isp_video_buffer_prepare_vm_flags - Get VMA flags for a userspace address | 408 | * isp_video_buffer_prepare_vm_flags - Get VMA flags for a userspace address |
| 409 | * | 409 | * |
| 410 | * This function locates the VMAs for the buffer's userspace address and checks | 410 | * This function locates the VMAs for the buffer's userspace address and checks |
| 411 | * that their flags match. The onlflag that we need to care for at the moment is | 411 | * that their flags match. The only flag that we need to care for at the moment |
| 412 | * VM_PFNMAP. | 412 | * is VM_PFNMAP. |
| 413 | * | 413 | * |
| 414 | * The buffer vm_flags field is set to the first VMA flags. | 414 | * The buffer vm_flags field is set to the first VMA flags. |
| 415 | * | 415 | * |
diff --git a/drivers/media/video/omap3isp/ispresizer.c b/drivers/media/video/omap3isp/ispresizer.c index 653f88ba56db..0bb0f8cd36f5 100644 --- a/drivers/media/video/omap3isp/ispresizer.c +++ b/drivers/media/video/omap3isp/ispresizer.c | |||
| @@ -714,19 +714,50 @@ static void resizer_print_status(struct isp_res_device *res) | |||
| 714 | * iw and ih are the input width and height after cropping. Those equations need | 714 | * iw and ih are the input width and height after cropping. Those equations need |
| 715 | * to be satisfied exactly for the resizer to work correctly. | 715 | * to be satisfied exactly for the resizer to work correctly. |
| 716 | * | 716 | * |
| 717 | * Reverting the equations, we can compute the resizing ratios with | 717 | * The equations can't be easily reverted, as the >> 8 operation is not linear. |
| 718 | * In addition, not all input sizes can be achieved for a given output size. To | ||
| 719 | * get the highest input size lower than or equal to the requested input size, | ||
| 720 | * we need to compute the highest resizing ratio that satisfies the following | ||
| 721 | * inequality (taking the 4-tap mode width equation as an example) | ||
| 722 | * | ||
| 723 | * iw >= (32 * sph + (ow - 1) * hrsz + 16) >> 8 - 7 | ||
| 724 | * | ||
| 725 | * (where iw is the requested input width) which can be rewritten as | ||
| 726 | * | ||
| 727 | * iw - 7 >= (32 * sph + (ow - 1) * hrsz + 16) >> 8 | ||
| 728 | * (iw - 7) << 8 >= 32 * sph + (ow - 1) * hrsz + 16 - b | ||
| 729 | * ((iw - 7) << 8) + b >= 32 * sph + (ow - 1) * hrsz + 16 | ||
| 730 | * | ||
| 731 | * where b is the value of the 8 least significant bits of the right hand side | ||
| 732 | * expression of the last inequality. The highest resizing ratio value will be | ||
| 733 | * achieved when b is equal to its maximum value of 255. That resizing ratio | ||
| 734 | * value will still satisfy the original inequality, as b will disappear when | ||
| 735 | * the expression will be shifted right by 8. | ||
| 736 | * | ||
| 737 | * The reverted the equations thus become | ||
| 718 | * | 738 | * |
| 719 | * - 8-phase, 4-tap mode | 739 | * - 8-phase, 4-tap mode |
| 720 | * hrsz = ((iw - 7) * 256 - 16 - 32 * sph) / (ow - 1) | 740 | * hrsz = ((iw - 7) * 256 + 255 - 16 - 32 * sph) / (ow - 1) |
| 721 | * vrsz = ((ih - 4) * 256 - 16 - 32 * spv) / (oh - 1) | 741 | * vrsz = ((ih - 4) * 256 + 255 - 16 - 32 * spv) / (oh - 1) |
| 722 | * - 4-phase, 7-tap mode | 742 | * - 4-phase, 7-tap mode |
| 723 | * hrsz = ((iw - 7) * 256 - 32 - 64 * sph) / (ow - 1) | 743 | * hrsz = ((iw - 7) * 256 + 255 - 32 - 64 * sph) / (ow - 1) |
| 724 | * vrsz = ((ih - 7) * 256 - 32 - 64 * spv) / (oh - 1) | 744 | * vrsz = ((ih - 7) * 256 + 255 - 32 - 64 * spv) / (oh - 1) |
| 725 | * | 745 | * |
| 726 | * The ratios are integer values, and must be rounded down to ensure that the | 746 | * The ratios are integer values, and are rounded down to ensure that the |
| 727 | * cropped input size is not bigger than the uncropped input size. As the ratio | 747 | * cropped input size is not bigger than the uncropped input size. |
| 728 | * in 7-tap mode is always smaller than the ratio in 4-tap mode, we can use the | 748 | * |
| 729 | * 7-tap mode equations to compute a ratio approximation. | 749 | * As the number of phases/taps, used to select the correct equations to compute |
| 750 | * the ratio, depends on the ratio, we start with the 4-tap mode equations to | ||
| 751 | * compute an approximation of the ratio, and switch to the 7-tap mode equations | ||
| 752 | * if the approximation is higher than the ratio threshold. | ||
| 753 | * | ||
| 754 | * As the 7-tap mode equations will return a ratio smaller than or equal to the | ||
| 755 | * 4-tap mode equations, the resulting ratio could become lower than or equal to | ||
| 756 | * the ratio threshold. This 'equations loop' isn't an issue as long as the | ||
| 757 | * correct equations are used to compute the final input size. Starting with the | ||
| 758 | * 4-tap mode equations ensure that, in case of values resulting in a 'ratio | ||
| 759 | * loop', the smallest of the ratio values will be used, never exceeding the | ||
| 760 | * requested input size. | ||
| 730 | * | 761 | * |
| 731 | * We first clamp the output size according to the hardware capabilitie to avoid | 762 | * We first clamp the output size according to the hardware capabilitie to avoid |
| 732 | * auto-cropping the input more than required to satisfy the TRM equations. The | 763 | * auto-cropping the input more than required to satisfy the TRM equations. The |
| @@ -775,6 +806,8 @@ static void resizer_calc_ratios(struct isp_res_device *res, | |||
| 775 | unsigned int max_width; | 806 | unsigned int max_width; |
| 776 | unsigned int max_height; | 807 | unsigned int max_height; |
| 777 | unsigned int width_alignment; | 808 | unsigned int width_alignment; |
| 809 | unsigned int width; | ||
| 810 | unsigned int height; | ||
| 778 | 811 | ||
| 779 | /* | 812 | /* |
| 780 | * Clamp the output height based on the hardware capabilities and | 813 | * Clamp the output height based on the hardware capabilities and |
| @@ -786,19 +819,22 @@ static void resizer_calc_ratios(struct isp_res_device *res, | |||
| 786 | max_height = min_t(unsigned int, max_height, MAX_OUT_HEIGHT); | 819 | max_height = min_t(unsigned int, max_height, MAX_OUT_HEIGHT); |
| 787 | output->height = clamp(output->height, min_height, max_height); | 820 | output->height = clamp(output->height, min_height, max_height); |
| 788 | 821 | ||
| 789 | ratio->vert = ((input->height - 7) * 256 - 32 - 64 * spv) | 822 | ratio->vert = ((input->height - 4) * 256 + 255 - 16 - 32 * spv) |
| 790 | / (output->height - 1); | 823 | / (output->height - 1); |
| 824 | if (ratio->vert > MID_RESIZE_VALUE) | ||
| 825 | ratio->vert = ((input->height - 7) * 256 + 255 - 32 - 64 * spv) | ||
| 826 | / (output->height - 1); | ||
| 791 | ratio->vert = clamp_t(unsigned int, ratio->vert, | 827 | ratio->vert = clamp_t(unsigned int, ratio->vert, |
| 792 | MIN_RESIZE_VALUE, MAX_RESIZE_VALUE); | 828 | MIN_RESIZE_VALUE, MAX_RESIZE_VALUE); |
| 793 | 829 | ||
| 794 | if (ratio->vert <= MID_RESIZE_VALUE) { | 830 | if (ratio->vert <= MID_RESIZE_VALUE) { |
| 795 | upscaled_height = (output->height - 1) * ratio->vert | 831 | upscaled_height = (output->height - 1) * ratio->vert |
| 796 | + 32 * spv + 16; | 832 | + 32 * spv + 16; |
| 797 | input->height = (upscaled_height >> 8) + 4; | 833 | height = (upscaled_height >> 8) + 4; |
| 798 | } else { | 834 | } else { |
| 799 | upscaled_height = (output->height - 1) * ratio->vert | 835 | upscaled_height = (output->height - 1) * ratio->vert |
| 800 | + 64 * spv + 32; | 836 | + 64 * spv + 32; |
| 801 | input->height = (upscaled_height >> 8) + 7; | 837 | height = (upscaled_height >> 8) + 7; |
| 802 | } | 838 | } |
| 803 | 839 | ||
| 804 | /* | 840 | /* |
| @@ -854,20 +890,29 @@ static void resizer_calc_ratios(struct isp_res_device *res, | |||
| 854 | max_width & ~(width_alignment - 1)); | 890 | max_width & ~(width_alignment - 1)); |
| 855 | output->width = ALIGN(output->width, width_alignment); | 891 | output->width = ALIGN(output->width, width_alignment); |
| 856 | 892 | ||
| 857 | ratio->horz = ((input->width - 7) * 256 - 32 - 64 * sph) | 893 | ratio->horz = ((input->width - 7) * 256 + 255 - 16 - 32 * sph) |
| 858 | / (output->width - 1); | 894 | / (output->width - 1); |
| 895 | if (ratio->horz > MID_RESIZE_VALUE) | ||
| 896 | ratio->horz = ((input->width - 7) * 256 + 255 - 32 - 64 * sph) | ||
| 897 | / (output->width - 1); | ||
| 859 | ratio->horz = clamp_t(unsigned int, ratio->horz, | 898 | ratio->horz = clamp_t(unsigned int, ratio->horz, |
| 860 | MIN_RESIZE_VALUE, MAX_RESIZE_VALUE); | 899 | MIN_RESIZE_VALUE, MAX_RESIZE_VALUE); |
| 861 | 900 | ||
| 862 | if (ratio->horz <= MID_RESIZE_VALUE) { | 901 | if (ratio->horz <= MID_RESIZE_VALUE) { |
| 863 | upscaled_width = (output->width - 1) * ratio->horz | 902 | upscaled_width = (output->width - 1) * ratio->horz |
| 864 | + 32 * sph + 16; | 903 | + 32 * sph + 16; |
| 865 | input->width = (upscaled_width >> 8) + 7; | 904 | width = (upscaled_width >> 8) + 7; |
| 866 | } else { | 905 | } else { |
| 867 | upscaled_width = (output->width - 1) * ratio->horz | 906 | upscaled_width = (output->width - 1) * ratio->horz |
| 868 | + 64 * sph + 32; | 907 | + 64 * sph + 32; |
| 869 | input->width = (upscaled_width >> 8) + 7; | 908 | width = (upscaled_width >> 8) + 7; |
| 870 | } | 909 | } |
| 910 | |||
| 911 | /* Center the new crop rectangle. */ | ||
| 912 | input->left += (input->width - width) / 2; | ||
| 913 | input->top += (input->height - height) / 2; | ||
| 914 | input->width = width; | ||
| 915 | input->height = height; | ||
| 871 | } | 916 | } |
| 872 | 917 | ||
| 873 | /* | 918 | /* |
diff --git a/drivers/media/video/omap3isp/ispstat.h b/drivers/media/video/omap3isp/ispstat.h index 820950c9ef46..d86da94fa50d 100644 --- a/drivers/media/video/omap3isp/ispstat.h +++ b/drivers/media/video/omap3isp/ispstat.h | |||
| @@ -131,9 +131,9 @@ struct ispstat { | |||
| 131 | struct ispstat_generic_config { | 131 | struct ispstat_generic_config { |
| 132 | /* | 132 | /* |
| 133 | * Fields must be in the same order as in: | 133 | * Fields must be in the same order as in: |
| 134 | * - isph3a_aewb_config | 134 | * - omap3isp_h3a_aewb_config |
| 135 | * - isph3a_af_config | 135 | * - omap3isp_h3a_af_config |
| 136 | * - isphist_config | 136 | * - omap3isp_hist_config |
| 137 | */ | 137 | */ |
| 138 | u32 buf_size; | 138 | u32 buf_size; |
| 139 | u16 config_counter; | 139 | u16 config_counter; |
diff --git a/drivers/media/video/omap3isp/ispvideo.c b/drivers/media/video/omap3isp/ispvideo.c index 208a7ec739d7..9cd8f1aa567b 100644 --- a/drivers/media/video/omap3isp/ispvideo.c +++ b/drivers/media/video/omap3isp/ispvideo.c | |||
| @@ -47,29 +47,59 @@ | |||
| 47 | 47 | ||
| 48 | static struct isp_format_info formats[] = { | 48 | static struct isp_format_info formats[] = { |
| 49 | { V4L2_MBUS_FMT_Y8_1X8, V4L2_MBUS_FMT_Y8_1X8, | 49 | { V4L2_MBUS_FMT_Y8_1X8, V4L2_MBUS_FMT_Y8_1X8, |
| 50 | V4L2_MBUS_FMT_Y8_1X8, V4L2_PIX_FMT_GREY, 8, }, | 50 | V4L2_MBUS_FMT_Y8_1X8, V4L2_MBUS_FMT_Y8_1X8, |
| 51 | V4L2_PIX_FMT_GREY, 8, }, | ||
| 52 | { V4L2_MBUS_FMT_Y10_1X10, V4L2_MBUS_FMT_Y10_1X10, | ||
| 53 | V4L2_MBUS_FMT_Y10_1X10, V4L2_MBUS_FMT_Y8_1X8, | ||
| 54 | V4L2_PIX_FMT_Y10, 10, }, | ||
| 55 | { V4L2_MBUS_FMT_Y12_1X12, V4L2_MBUS_FMT_Y10_1X10, | ||
| 56 | V4L2_MBUS_FMT_Y12_1X12, V4L2_MBUS_FMT_Y8_1X8, | ||
| 57 | V4L2_PIX_FMT_Y12, 12, }, | ||
| 58 | { V4L2_MBUS_FMT_SBGGR8_1X8, V4L2_MBUS_FMT_SBGGR8_1X8, | ||
| 59 | V4L2_MBUS_FMT_SBGGR8_1X8, V4L2_MBUS_FMT_SBGGR8_1X8, | ||
| 60 | V4L2_PIX_FMT_SBGGR8, 8, }, | ||
| 61 | { V4L2_MBUS_FMT_SGBRG8_1X8, V4L2_MBUS_FMT_SGBRG8_1X8, | ||
| 62 | V4L2_MBUS_FMT_SGBRG8_1X8, V4L2_MBUS_FMT_SGBRG8_1X8, | ||
| 63 | V4L2_PIX_FMT_SGBRG8, 8, }, | ||
| 64 | { V4L2_MBUS_FMT_SGRBG8_1X8, V4L2_MBUS_FMT_SGRBG8_1X8, | ||
| 65 | V4L2_MBUS_FMT_SGRBG8_1X8, V4L2_MBUS_FMT_SGRBG8_1X8, | ||
| 66 | V4L2_PIX_FMT_SGRBG8, 8, }, | ||
| 67 | { V4L2_MBUS_FMT_SRGGB8_1X8, V4L2_MBUS_FMT_SRGGB8_1X8, | ||
| 68 | V4L2_MBUS_FMT_SRGGB8_1X8, V4L2_MBUS_FMT_SRGGB8_1X8, | ||
| 69 | V4L2_PIX_FMT_SRGGB8, 8, }, | ||
| 51 | { V4L2_MBUS_FMT_SGRBG10_DPCM8_1X8, V4L2_MBUS_FMT_SGRBG10_DPCM8_1X8, | 70 | { V4L2_MBUS_FMT_SGRBG10_DPCM8_1X8, V4L2_MBUS_FMT_SGRBG10_DPCM8_1X8, |
| 52 | V4L2_MBUS_FMT_SGRBG10_1X10, V4L2_PIX_FMT_SGRBG10DPCM8, 8, }, | 71 | V4L2_MBUS_FMT_SGRBG10_1X10, 0, |
| 72 | V4L2_PIX_FMT_SGRBG10DPCM8, 8, }, | ||
| 53 | { V4L2_MBUS_FMT_SBGGR10_1X10, V4L2_MBUS_FMT_SBGGR10_1X10, | 73 | { V4L2_MBUS_FMT_SBGGR10_1X10, V4L2_MBUS_FMT_SBGGR10_1X10, |
| 54 | V4L2_MBUS_FMT_SBGGR10_1X10, V4L2_PIX_FMT_SBGGR10, 10, }, | 74 | V4L2_MBUS_FMT_SBGGR10_1X10, V4L2_MBUS_FMT_SBGGR8_1X8, |
| 75 | V4L2_PIX_FMT_SBGGR10, 10, }, | ||
| 55 | { V4L2_MBUS_FMT_SGBRG10_1X10, V4L2_MBUS_FMT_SGBRG10_1X10, | 76 | { V4L2_MBUS_FMT_SGBRG10_1X10, V4L2_MBUS_FMT_SGBRG10_1X10, |
| 56 | V4L2_MBUS_FMT_SGBRG10_1X10, V4L2_PIX_FMT_SGBRG10, 10, }, | 77 | V4L2_MBUS_FMT_SGBRG10_1X10, V4L2_MBUS_FMT_SGBRG8_1X8, |
| 78 | V4L2_PIX_FMT_SGBRG10, 10, }, | ||
| 57 | { V4L2_MBUS_FMT_SGRBG10_1X10, V4L2_MBUS_FMT_SGRBG10_1X10, | 79 | { V4L2_MBUS_FMT_SGRBG10_1X10, V4L2_MBUS_FMT_SGRBG10_1X10, |
| 58 | V4L2_MBUS_FMT_SGRBG10_1X10, V4L2_PIX_FMT_SGRBG10, 10, }, | 80 | V4L2_MBUS_FMT_SGRBG10_1X10, V4L2_MBUS_FMT_SGRBG8_1X8, |
| 81 | V4L2_PIX_FMT_SGRBG10, 10, }, | ||
| 59 | { V4L2_MBUS_FMT_SRGGB10_1X10, V4L2_MBUS_FMT_SRGGB10_1X10, | 82 | { V4L2_MBUS_FMT_SRGGB10_1X10, V4L2_MBUS_FMT_SRGGB10_1X10, |
| 60 | V4L2_MBUS_FMT_SRGGB10_1X10, V4L2_PIX_FMT_SRGGB10, 10, }, | 83 | V4L2_MBUS_FMT_SRGGB10_1X10, V4L2_MBUS_FMT_SRGGB8_1X8, |
| 84 | V4L2_PIX_FMT_SRGGB10, 10, }, | ||
| 61 | { V4L2_MBUS_FMT_SBGGR12_1X12, V4L2_MBUS_FMT_SBGGR10_1X10, | 85 | { V4L2_MBUS_FMT_SBGGR12_1X12, V4L2_MBUS_FMT_SBGGR10_1X10, |
| 62 | V4L2_MBUS_FMT_SBGGR12_1X12, V4L2_PIX_FMT_SBGGR12, 12, }, | 86 | V4L2_MBUS_FMT_SBGGR12_1X12, V4L2_MBUS_FMT_SBGGR8_1X8, |
| 87 | V4L2_PIX_FMT_SBGGR12, 12, }, | ||
| 63 | { V4L2_MBUS_FMT_SGBRG12_1X12, V4L2_MBUS_FMT_SGBRG10_1X10, | 88 | { V4L2_MBUS_FMT_SGBRG12_1X12, V4L2_MBUS_FMT_SGBRG10_1X10, |
| 64 | V4L2_MBUS_FMT_SGBRG12_1X12, V4L2_PIX_FMT_SGBRG12, 12, }, | 89 | V4L2_MBUS_FMT_SGBRG12_1X12, V4L2_MBUS_FMT_SGBRG8_1X8, |
| 90 | V4L2_PIX_FMT_SGBRG12, 12, }, | ||
| 65 | { V4L2_MBUS_FMT_SGRBG12_1X12, V4L2_MBUS_FMT_SGRBG10_1X10, | 91 | { V4L2_MBUS_FMT_SGRBG12_1X12, V4L2_MBUS_FMT_SGRBG10_1X10, |
| 66 | V4L2_MBUS_FMT_SGRBG12_1X12, V4L2_PIX_FMT_SGRBG12, 12, }, | 92 | V4L2_MBUS_FMT_SGRBG12_1X12, V4L2_MBUS_FMT_SGRBG8_1X8, |
| 93 | V4L2_PIX_FMT_SGRBG12, 12, }, | ||
| 67 | { V4L2_MBUS_FMT_SRGGB12_1X12, V4L2_MBUS_FMT_SRGGB10_1X10, | 94 | { V4L2_MBUS_FMT_SRGGB12_1X12, V4L2_MBUS_FMT_SRGGB10_1X10, |
| 68 | V4L2_MBUS_FMT_SRGGB12_1X12, V4L2_PIX_FMT_SRGGB12, 12, }, | 95 | V4L2_MBUS_FMT_SRGGB12_1X12, V4L2_MBUS_FMT_SRGGB8_1X8, |
| 96 | V4L2_PIX_FMT_SRGGB12, 12, }, | ||
| 69 | { V4L2_MBUS_FMT_UYVY8_1X16, V4L2_MBUS_FMT_UYVY8_1X16, | 97 | { V4L2_MBUS_FMT_UYVY8_1X16, V4L2_MBUS_FMT_UYVY8_1X16, |
| 70 | V4L2_MBUS_FMT_UYVY8_1X16, V4L2_PIX_FMT_UYVY, 16, }, | 98 | V4L2_MBUS_FMT_UYVY8_1X16, 0, |
| 99 | V4L2_PIX_FMT_UYVY, 16, }, | ||
| 71 | { V4L2_MBUS_FMT_YUYV8_1X16, V4L2_MBUS_FMT_YUYV8_1X16, | 100 | { V4L2_MBUS_FMT_YUYV8_1X16, V4L2_MBUS_FMT_YUYV8_1X16, |
| 72 | V4L2_MBUS_FMT_YUYV8_1X16, V4L2_PIX_FMT_YUYV, 16, }, | 101 | V4L2_MBUS_FMT_YUYV8_1X16, 0, |
| 102 | V4L2_PIX_FMT_YUYV, 16, }, | ||
| 73 | }; | 103 | }; |
| 74 | 104 | ||
| 75 | const struct isp_format_info * | 105 | const struct isp_format_info * |
| @@ -86,6 +116,37 @@ omap3isp_video_format_info(enum v4l2_mbus_pixelcode code) | |||
| 86 | } | 116 | } |
| 87 | 117 | ||
| 88 | /* | 118 | /* |
| 119 | * Decide whether desired output pixel code can be obtained with | ||
| 120 | * the lane shifter by shifting the input pixel code. | ||
| 121 | * @in: input pixelcode to shifter | ||
| 122 | * @out: output pixelcode from shifter | ||
| 123 | * @additional_shift: # of bits the sensor's LSB is offset from CAMEXT[0] | ||
| 124 | * | ||
| 125 | * return true if the combination is possible | ||
| 126 | * return false otherwise | ||
| 127 | */ | ||
| 128 | static bool isp_video_is_shiftable(enum v4l2_mbus_pixelcode in, | ||
| 129 | enum v4l2_mbus_pixelcode out, | ||
| 130 | unsigned int additional_shift) | ||
| 131 | { | ||
| 132 | const struct isp_format_info *in_info, *out_info; | ||
| 133 | |||
| 134 | if (in == out) | ||
| 135 | return true; | ||
| 136 | |||
| 137 | in_info = omap3isp_video_format_info(in); | ||
| 138 | out_info = omap3isp_video_format_info(out); | ||
| 139 | |||
| 140 | if ((in_info->flavor == 0) || (out_info->flavor == 0)) | ||
| 141 | return false; | ||
| 142 | |||
| 143 | if (in_info->flavor != out_info->flavor) | ||
| 144 | return false; | ||
| 145 | |||
| 146 | return in_info->bpp - out_info->bpp + additional_shift <= 6; | ||
| 147 | } | ||
| 148 | |||
| 149 | /* | ||
| 89 | * isp_video_mbus_to_pix - Convert v4l2_mbus_framefmt to v4l2_pix_format | 150 | * isp_video_mbus_to_pix - Convert v4l2_mbus_framefmt to v4l2_pix_format |
| 90 | * @video: ISP video instance | 151 | * @video: ISP video instance |
| 91 | * @mbus: v4l2_mbus_framefmt format (input) | 152 | * @mbus: v4l2_mbus_framefmt format (input) |
| @@ -235,6 +296,7 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe) | |||
| 235 | return -EPIPE; | 296 | return -EPIPE; |
| 236 | 297 | ||
| 237 | while (1) { | 298 | while (1) { |
| 299 | unsigned int shifter_link; | ||
| 238 | /* Retrieve the sink format */ | 300 | /* Retrieve the sink format */ |
| 239 | pad = &subdev->entity.pads[0]; | 301 | pad = &subdev->entity.pads[0]; |
| 240 | if (!(pad->flags & MEDIA_PAD_FL_SINK)) | 302 | if (!(pad->flags & MEDIA_PAD_FL_SINK)) |
| @@ -263,6 +325,10 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe) | |||
| 263 | return -ENOSPC; | 325 | return -ENOSPC; |
| 264 | } | 326 | } |
| 265 | 327 | ||
| 328 | /* If sink pad is on CCDC, the link has the lane shifter | ||
| 329 | * in the middle of it. */ | ||
| 330 | shifter_link = subdev == &isp->isp_ccdc.subdev; | ||
| 331 | |||
| 266 | /* Retrieve the source format */ | 332 | /* Retrieve the source format */ |
| 267 | pad = media_entity_remote_source(pad); | 333 | pad = media_entity_remote_source(pad); |
| 268 | if (pad == NULL || | 334 | if (pad == NULL || |
| @@ -278,10 +344,24 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe) | |||
| 278 | return -EPIPE; | 344 | return -EPIPE; |
| 279 | 345 | ||
| 280 | /* Check if the two ends match */ | 346 | /* Check if the two ends match */ |
| 281 | if (fmt_source.format.code != fmt_sink.format.code || | 347 | if (fmt_source.format.width != fmt_sink.format.width || |
| 282 | fmt_source.format.width != fmt_sink.format.width || | ||
| 283 | fmt_source.format.height != fmt_sink.format.height) | 348 | fmt_source.format.height != fmt_sink.format.height) |
| 284 | return -EPIPE; | 349 | return -EPIPE; |
| 350 | |||
| 351 | if (shifter_link) { | ||
| 352 | unsigned int parallel_shift = 0; | ||
| 353 | if (isp->isp_ccdc.input == CCDC_INPUT_PARALLEL) { | ||
| 354 | struct isp_parallel_platform_data *pdata = | ||
| 355 | &((struct isp_v4l2_subdevs_group *) | ||
| 356 | subdev->host_priv)->bus.parallel; | ||
| 357 | parallel_shift = pdata->data_lane_shift * 2; | ||
| 358 | } | ||
| 359 | if (!isp_video_is_shiftable(fmt_source.format.code, | ||
| 360 | fmt_sink.format.code, | ||
| 361 | parallel_shift)) | ||
| 362 | return -EPIPE; | ||
| 363 | } else if (fmt_source.format.code != fmt_sink.format.code) | ||
| 364 | return -EPIPE; | ||
| 285 | } | 365 | } |
| 286 | 366 | ||
| 287 | return 0; | 367 | return 0; |
diff --git a/drivers/media/video/omap3isp/ispvideo.h b/drivers/media/video/omap3isp/ispvideo.h index 524a1acd0906..911bea64e78a 100644 --- a/drivers/media/video/omap3isp/ispvideo.h +++ b/drivers/media/video/omap3isp/ispvideo.h | |||
| @@ -49,6 +49,8 @@ struct v4l2_pix_format; | |||
| 49 | * bits. Identical to @code if the format is 10 bits wide or less. | 49 | * bits. Identical to @code if the format is 10 bits wide or less. |
| 50 | * @uncompressed: V4L2 media bus format code for the corresponding uncompressed | 50 | * @uncompressed: V4L2 media bus format code for the corresponding uncompressed |
| 51 | * format. Identical to @code if the format is not DPCM compressed. | 51 | * format. Identical to @code if the format is not DPCM compressed. |
| 52 | * @flavor: V4L2 media bus format code for the same pixel layout but | ||
| 53 | * shifted to be 8 bits per pixel. =0 if format is not shiftable. | ||
| 52 | * @pixelformat: V4L2 pixel format FCC identifier | 54 | * @pixelformat: V4L2 pixel format FCC identifier |
| 53 | * @bpp: Bits per pixel | 55 | * @bpp: Bits per pixel |
| 54 | */ | 56 | */ |
| @@ -56,6 +58,7 @@ struct isp_format_info { | |||
| 56 | enum v4l2_mbus_pixelcode code; | 58 | enum v4l2_mbus_pixelcode code; |
| 57 | enum v4l2_mbus_pixelcode truncated; | 59 | enum v4l2_mbus_pixelcode truncated; |
| 58 | enum v4l2_mbus_pixelcode uncompressed; | 60 | enum v4l2_mbus_pixelcode uncompressed; |
| 61 | enum v4l2_mbus_pixelcode flavor; | ||
| 59 | u32 pixelformat; | 62 | u32 pixelformat; |
| 60 | unsigned int bpp; | 63 | unsigned int bpp; |
| 61 | }; | 64 | }; |
diff --git a/drivers/media/video/s5p-fimc/fimc-capture.c b/drivers/media/video/s5p-fimc/fimc-capture.c index 95f8b4e11e46..d142b40ea64e 100644 --- a/drivers/media/video/s5p-fimc/fimc-capture.c +++ b/drivers/media/video/s5p-fimc/fimc-capture.c | |||
| @@ -527,7 +527,7 @@ static int fimc_cap_s_fmt_mplane(struct file *file, void *priv, | |||
| 527 | if (ret) | 527 | if (ret) |
| 528 | return ret; | 528 | return ret; |
| 529 | 529 | ||
| 530 | if (vb2_is_streaming(&fimc->vid_cap.vbq) || fimc_capture_active(fimc)) | 530 | if (vb2_is_busy(&fimc->vid_cap.vbq) || fimc_capture_active(fimc)) |
| 531 | return -EBUSY; | 531 | return -EBUSY; |
| 532 | 532 | ||
| 533 | frame = &ctx->d_frame; | 533 | frame = &ctx->d_frame; |
| @@ -539,8 +539,10 @@ static int fimc_cap_s_fmt_mplane(struct file *file, void *priv, | |||
| 539 | return -EINVAL; | 539 | return -EINVAL; |
| 540 | } | 540 | } |
| 541 | 541 | ||
| 542 | for (i = 0; i < frame->fmt->colplanes; i++) | 542 | for (i = 0; i < frame->fmt->colplanes; i++) { |
| 543 | frame->payload[i] = pix->plane_fmt[i].bytesperline * pix->height; | 543 | frame->payload[i] = |
| 544 | (pix->width * pix->height * frame->fmt->depth[i]) >> 3; | ||
| 545 | } | ||
| 544 | 546 | ||
| 545 | /* Output DMA frame pixel size and offsets. */ | 547 | /* Output DMA frame pixel size and offsets. */ |
| 546 | frame->f_width = pix->plane_fmt[0].bytesperline * 8 | 548 | frame->f_width = pix->plane_fmt[0].bytesperline * 8 |
diff --git a/drivers/media/video/s5p-fimc/fimc-core.c b/drivers/media/video/s5p-fimc/fimc-core.c index 6c919b38a3d8..dc91a8511af6 100644 --- a/drivers/media/video/s5p-fimc/fimc-core.c +++ b/drivers/media/video/s5p-fimc/fimc-core.c | |||
| @@ -361,10 +361,20 @@ static void fimc_capture_irq_handler(struct fimc_dev *fimc) | |||
| 361 | { | 361 | { |
| 362 | struct fimc_vid_cap *cap = &fimc->vid_cap; | 362 | struct fimc_vid_cap *cap = &fimc->vid_cap; |
| 363 | struct fimc_vid_buffer *v_buf; | 363 | struct fimc_vid_buffer *v_buf; |
| 364 | struct timeval *tv; | ||
| 365 | struct timespec ts; | ||
| 364 | 366 | ||
| 365 | if (!list_empty(&cap->active_buf_q) && | 367 | if (!list_empty(&cap->active_buf_q) && |
| 366 | test_bit(ST_CAPT_RUN, &fimc->state)) { | 368 | test_bit(ST_CAPT_RUN, &fimc->state)) { |
| 369 | ktime_get_real_ts(&ts); | ||
| 370 | |||
| 367 | v_buf = active_queue_pop(cap); | 371 | v_buf = active_queue_pop(cap); |
| 372 | |||
| 373 | tv = &v_buf->vb.v4l2_buf.timestamp; | ||
| 374 | tv->tv_sec = ts.tv_sec; | ||
| 375 | tv->tv_usec = ts.tv_nsec / NSEC_PER_USEC; | ||
| 376 | v_buf->vb.v4l2_buf.sequence = cap->frame_count++; | ||
| 377 | |||
| 368 | vb2_buffer_done(&v_buf->vb, VB2_BUF_STATE_DONE); | 378 | vb2_buffer_done(&v_buf->vb, VB2_BUF_STATE_DONE); |
| 369 | } | 379 | } |
| 370 | 380 | ||
| @@ -758,7 +768,7 @@ static void fimc_unlock(struct vb2_queue *vq) | |||
| 758 | mutex_unlock(&ctx->fimc_dev->lock); | 768 | mutex_unlock(&ctx->fimc_dev->lock); |
| 759 | } | 769 | } |
| 760 | 770 | ||
| 761 | struct vb2_ops fimc_qops = { | 771 | static struct vb2_ops fimc_qops = { |
| 762 | .queue_setup = fimc_queue_setup, | 772 | .queue_setup = fimc_queue_setup, |
| 763 | .buf_prepare = fimc_buf_prepare, | 773 | .buf_prepare = fimc_buf_prepare, |
| 764 | .buf_queue = fimc_buf_queue, | 774 | .buf_queue = fimc_buf_queue, |
| @@ -927,23 +937,23 @@ int fimc_vidioc_try_fmt_mplane(struct file *file, void *priv, | |||
| 927 | pix->num_planes = fmt->memplanes; | 937 | pix->num_planes = fmt->memplanes; |
| 928 | pix->colorspace = V4L2_COLORSPACE_JPEG; | 938 | pix->colorspace = V4L2_COLORSPACE_JPEG; |
| 929 | 939 | ||
| 930 | for (i = 0; i < pix->num_planes; ++i) { | ||
| 931 | int bpl = pix->plane_fmt[i].bytesperline; | ||
| 932 | 940 | ||
| 933 | dbg("[%d] bpl: %d, depth: %d, w: %d, h: %d", | 941 | for (i = 0; i < pix->num_planes; ++i) { |
| 934 | i, bpl, fmt->depth[i], pix->width, pix->height); | 942 | u32 bpl = pix->plane_fmt[i].bytesperline; |
| 943 | u32 *sizeimage = &pix->plane_fmt[i].sizeimage; | ||
| 935 | 944 | ||
| 936 | if (!bpl || (bpl * 8 / fmt->depth[i]) > pix->width) | 945 | if (fmt->colplanes > 1 && (bpl == 0 || bpl < pix->width)) |
| 937 | bpl = (pix->width * fmt->depth[0]) >> 3; | 946 | bpl = pix->width; /* Planar */ |
| 938 | 947 | ||
| 939 | if (!pix->plane_fmt[i].sizeimage) | 948 | if (fmt->colplanes == 1 && /* Packed */ |
| 940 | pix->plane_fmt[i].sizeimage = pix->height * bpl; | 949 | (bpl == 0 || ((bpl * 8) / fmt->depth[i]) < pix->width)) |
| 950 | bpl = (pix->width * fmt->depth[0]) / 8; | ||
| 941 | 951 | ||
| 942 | pix->plane_fmt[i].bytesperline = bpl; | 952 | if (i == 0) /* Same bytesperline for each plane. */ |
| 953 | mod_x = bpl; | ||
| 943 | 954 | ||
| 944 | dbg("[%d]: bpl: %d, sizeimage: %d", | 955 | pix->plane_fmt[i].bytesperline = mod_x; |
| 945 | i, pix->plane_fmt[i].bytesperline, | 956 | *sizeimage = (pix->width * pix->height * fmt->depth[i]) / 8; |
| 946 | pix->plane_fmt[i].sizeimage); | ||
| 947 | } | 957 | } |
| 948 | 958 | ||
| 949 | return 0; | 959 | return 0; |
| @@ -965,7 +975,7 @@ static int fimc_m2m_s_fmt_mplane(struct file *file, void *priv, | |||
| 965 | 975 | ||
| 966 | vq = v4l2_m2m_get_vq(ctx->m2m_ctx, f->type); | 976 | vq = v4l2_m2m_get_vq(ctx->m2m_ctx, f->type); |
| 967 | 977 | ||
| 968 | if (vb2_is_streaming(vq)) { | 978 | if (vb2_is_busy(vq)) { |
| 969 | v4l2_err(&fimc->m2m.v4l2_dev, "queue (%d) busy\n", f->type); | 979 | v4l2_err(&fimc->m2m.v4l2_dev, "queue (%d) busy\n", f->type); |
| 970 | return -EBUSY; | 980 | return -EBUSY; |
| 971 | } | 981 | } |
| @@ -985,8 +995,10 @@ static int fimc_m2m_s_fmt_mplane(struct file *file, void *priv, | |||
| 985 | if (!frame->fmt) | 995 | if (!frame->fmt) |
| 986 | return -EINVAL; | 996 | return -EINVAL; |
| 987 | 997 | ||
| 988 | for (i = 0; i < frame->fmt->colplanes; i++) | 998 | for (i = 0; i < frame->fmt->colplanes; i++) { |
| 989 | frame->payload[i] = pix->plane_fmt[i].bytesperline * pix->height; | 999 | frame->payload[i] = |
| 1000 | (pix->width * pix->height * frame->fmt->depth[i]) / 8; | ||
| 1001 | } | ||
| 990 | 1002 | ||
| 991 | frame->f_width = pix->plane_fmt[0].bytesperline * 8 / | 1003 | frame->f_width = pix->plane_fmt[0].bytesperline * 8 / |
| 992 | frame->fmt->depth[0]; | 1004 | frame->fmt->depth[0]; |
| @@ -1750,7 +1762,7 @@ static int __devexit fimc_remove(struct platform_device *pdev) | |||
| 1750 | } | 1762 | } |
| 1751 | 1763 | ||
| 1752 | /* Image pixel limits, similar across several FIMC HW revisions. */ | 1764 | /* Image pixel limits, similar across several FIMC HW revisions. */ |
| 1753 | static struct fimc_pix_limit s5p_pix_limit[3] = { | 1765 | static struct fimc_pix_limit s5p_pix_limit[4] = { |
| 1754 | [0] = { | 1766 | [0] = { |
| 1755 | .scaler_en_w = 3264, | 1767 | .scaler_en_w = 3264, |
| 1756 | .scaler_dis_w = 8192, | 1768 | .scaler_dis_w = 8192, |
| @@ -1775,6 +1787,14 @@ static struct fimc_pix_limit s5p_pix_limit[3] = { | |||
| 1775 | .out_rot_en_w = 1280, | 1787 | .out_rot_en_w = 1280, |
| 1776 | .out_rot_dis_w = 1920, | 1788 | .out_rot_dis_w = 1920, |
| 1777 | }, | 1789 | }, |
| 1790 | [3] = { | ||
| 1791 | .scaler_en_w = 1920, | ||
| 1792 | .scaler_dis_w = 8192, | ||
| 1793 | .in_rot_en_h = 1366, | ||
| 1794 | .in_rot_dis_w = 8192, | ||
| 1795 | .out_rot_en_w = 1366, | ||
| 1796 | .out_rot_dis_w = 1920, | ||
| 1797 | }, | ||
| 1778 | }; | 1798 | }; |
| 1779 | 1799 | ||
| 1780 | static struct samsung_fimc_variant fimc0_variant_s5p = { | 1800 | static struct samsung_fimc_variant fimc0_variant_s5p = { |
| @@ -1827,7 +1847,7 @@ static struct samsung_fimc_variant fimc2_variant_s5pv210 = { | |||
| 1827 | .pix_limit = &s5p_pix_limit[2], | 1847 | .pix_limit = &s5p_pix_limit[2], |
| 1828 | }; | 1848 | }; |
| 1829 | 1849 | ||
| 1830 | static struct samsung_fimc_variant fimc0_variant_s5pv310 = { | 1850 | static struct samsung_fimc_variant fimc0_variant_exynos4 = { |
| 1831 | .pix_hoff = 1, | 1851 | .pix_hoff = 1, |
| 1832 | .has_inp_rot = 1, | 1852 | .has_inp_rot = 1, |
| 1833 | .has_out_rot = 1, | 1853 | .has_out_rot = 1, |
| @@ -1840,7 +1860,7 @@ static struct samsung_fimc_variant fimc0_variant_s5pv310 = { | |||
| 1840 | .pix_limit = &s5p_pix_limit[1], | 1860 | .pix_limit = &s5p_pix_limit[1], |
| 1841 | }; | 1861 | }; |
| 1842 | 1862 | ||
| 1843 | static struct samsung_fimc_variant fimc2_variant_s5pv310 = { | 1863 | static struct samsung_fimc_variant fimc2_variant_exynos4 = { |
| 1844 | .pix_hoff = 1, | 1864 | .pix_hoff = 1, |
| 1845 | .has_cistatus2 = 1, | 1865 | .has_cistatus2 = 1, |
| 1846 | .has_mainscaler_ext = 1, | 1866 | .has_mainscaler_ext = 1, |
| @@ -1848,7 +1868,7 @@ static struct samsung_fimc_variant fimc2_variant_s5pv310 = { | |||
| 1848 | .min_out_pixsize = 16, | 1868 | .min_out_pixsize = 16, |
| 1849 | .hor_offs_align = 1, | 1869 | .hor_offs_align = 1, |
| 1850 | .out_buf_count = 32, | 1870 | .out_buf_count = 32, |
| 1851 | .pix_limit = &s5p_pix_limit[2], | 1871 | .pix_limit = &s5p_pix_limit[3], |
| 1852 | }; | 1872 | }; |
| 1853 | 1873 | ||
| 1854 | /* S5PC100 */ | 1874 | /* S5PC100 */ |
| @@ -1874,12 +1894,12 @@ static struct samsung_fimc_driverdata fimc_drvdata_s5pv210 = { | |||
| 1874 | }; | 1894 | }; |
| 1875 | 1895 | ||
| 1876 | /* S5PV310, S5PC210 */ | 1896 | /* S5PV310, S5PC210 */ |
| 1877 | static struct samsung_fimc_driverdata fimc_drvdata_s5pv310 = { | 1897 | static struct samsung_fimc_driverdata fimc_drvdata_exynos4 = { |
| 1878 | .variant = { | 1898 | .variant = { |
| 1879 | [0] = &fimc0_variant_s5pv310, | 1899 | [0] = &fimc0_variant_exynos4, |
| 1880 | [1] = &fimc0_variant_s5pv310, | 1900 | [1] = &fimc0_variant_exynos4, |
| 1881 | [2] = &fimc0_variant_s5pv310, | 1901 | [2] = &fimc0_variant_exynos4, |
| 1882 | [3] = &fimc2_variant_s5pv310, | 1902 | [3] = &fimc2_variant_exynos4, |
| 1883 | }, | 1903 | }, |
| 1884 | .num_entities = 4, | 1904 | .num_entities = 4, |
| 1885 | .lclk_frequency = 166000000UL, | 1905 | .lclk_frequency = 166000000UL, |
| @@ -1893,8 +1913,8 @@ static struct platform_device_id fimc_driver_ids[] = { | |||
| 1893 | .name = "s5pv210-fimc", | 1913 | .name = "s5pv210-fimc", |
| 1894 | .driver_data = (unsigned long)&fimc_drvdata_s5pv210, | 1914 | .driver_data = (unsigned long)&fimc_drvdata_s5pv210, |
| 1895 | }, { | 1915 | }, { |
| 1896 | .name = "s5pv310-fimc", | 1916 | .name = "exynos4-fimc", |
| 1897 | .driver_data = (unsigned long)&fimc_drvdata_s5pv310, | 1917 | .driver_data = (unsigned long)&fimc_drvdata_exynos4, |
| 1898 | }, | 1918 | }, |
| 1899 | {}, | 1919 | {}, |
| 1900 | }; | 1920 | }; |
diff --git a/drivers/media/video/sh_mobile_ceu_camera.c b/drivers/media/video/sh_mobile_ceu_camera.c index 3fe54bf41142..134e86bf6d97 100644 --- a/drivers/media/video/sh_mobile_ceu_camera.c +++ b/drivers/media/video/sh_mobile_ceu_camera.c | |||
| @@ -922,7 +922,7 @@ static int sh_mobile_ceu_get_formats(struct soc_camera_device *icd, unsigned int | |||
| 922 | /* Try 2560x1920, 1280x960, 640x480, 320x240 */ | 922 | /* Try 2560x1920, 1280x960, 640x480, 320x240 */ |
| 923 | mf.width = 2560 >> shift; | 923 | mf.width = 2560 >> shift; |
| 924 | mf.height = 1920 >> shift; | 924 | mf.height = 1920 >> shift; |
| 925 | ret = v4l2_device_call_until_err(sd->v4l2_dev, 0, video, | 925 | ret = v4l2_device_call_until_err(sd->v4l2_dev, (long)icd, video, |
| 926 | s_mbus_fmt, &mf); | 926 | s_mbus_fmt, &mf); |
| 927 | if (ret < 0) | 927 | if (ret < 0) |
| 928 | return ret; | 928 | return ret; |
| @@ -1224,7 +1224,7 @@ static int client_s_fmt(struct soc_camera_device *icd, | |||
| 1224 | struct v4l2_cropcap cap; | 1224 | struct v4l2_cropcap cap; |
| 1225 | int ret; | 1225 | int ret; |
| 1226 | 1226 | ||
| 1227 | ret = v4l2_device_call_until_err(sd->v4l2_dev, 0, video, | 1227 | ret = v4l2_device_call_until_err(sd->v4l2_dev, (long)icd, video, |
| 1228 | s_mbus_fmt, mf); | 1228 | s_mbus_fmt, mf); |
| 1229 | if (ret < 0) | 1229 | if (ret < 0) |
| 1230 | return ret; | 1230 | return ret; |
| @@ -1254,7 +1254,7 @@ static int client_s_fmt(struct soc_camera_device *icd, | |||
| 1254 | tmp_h = min(2 * tmp_h, max_height); | 1254 | tmp_h = min(2 * tmp_h, max_height); |
| 1255 | mf->width = tmp_w; | 1255 | mf->width = tmp_w; |
| 1256 | mf->height = tmp_h; | 1256 | mf->height = tmp_h; |
| 1257 | ret = v4l2_device_call_until_err(sd->v4l2_dev, 0, video, | 1257 | ret = v4l2_device_call_until_err(sd->v4l2_dev, (long)icd, video, |
| 1258 | s_mbus_fmt, mf); | 1258 | s_mbus_fmt, mf); |
| 1259 | dev_geo(dev, "Camera scaled to %ux%u\n", | 1259 | dev_geo(dev, "Camera scaled to %ux%u\n", |
| 1260 | mf->width, mf->height); | 1260 | mf->width, mf->height); |
| @@ -1658,7 +1658,7 @@ static int sh_mobile_ceu_try_fmt(struct soc_camera_device *icd, | |||
| 1658 | mf.code = xlate->code; | 1658 | mf.code = xlate->code; |
| 1659 | mf.colorspace = pix->colorspace; | 1659 | mf.colorspace = pix->colorspace; |
| 1660 | 1660 | ||
| 1661 | ret = v4l2_device_call_until_err(sd->v4l2_dev, 0, video, try_mbus_fmt, &mf); | 1661 | ret = v4l2_device_call_until_err(sd->v4l2_dev, (long)icd, video, try_mbus_fmt, &mf); |
| 1662 | if (ret < 0) | 1662 | if (ret < 0) |
| 1663 | return ret; | 1663 | return ret; |
| 1664 | 1664 | ||
| @@ -1682,7 +1682,7 @@ static int sh_mobile_ceu_try_fmt(struct soc_camera_device *icd, | |||
| 1682 | */ | 1682 | */ |
| 1683 | mf.width = 2560; | 1683 | mf.width = 2560; |
| 1684 | mf.height = 1920; | 1684 | mf.height = 1920; |
| 1685 | ret = v4l2_device_call_until_err(sd->v4l2_dev, 0, video, | 1685 | ret = v4l2_device_call_until_err(sd->v4l2_dev, (long)icd, video, |
| 1686 | try_mbus_fmt, &mf); | 1686 | try_mbus_fmt, &mf); |
| 1687 | if (ret < 0) { | 1687 | if (ret < 0) { |
| 1688 | /* Shouldn't actually happen... */ | 1688 | /* Shouldn't actually happen... */ |
diff --git a/drivers/media/video/sh_mobile_csi2.c b/drivers/media/video/sh_mobile_csi2.c index dd1b81b1442b..98b87481fa94 100644 --- a/drivers/media/video/sh_mobile_csi2.c +++ b/drivers/media/video/sh_mobile_csi2.c | |||
| @@ -38,6 +38,8 @@ struct sh_csi2 { | |||
| 38 | void __iomem *base; | 38 | void __iomem *base; |
| 39 | struct platform_device *pdev; | 39 | struct platform_device *pdev; |
| 40 | struct sh_csi2_client_config *client; | 40 | struct sh_csi2_client_config *client; |
| 41 | unsigned long (*query_bus_param)(struct soc_camera_device *); | ||
| 42 | int (*set_bus_param)(struct soc_camera_device *, unsigned long); | ||
| 41 | }; | 43 | }; |
| 42 | 44 | ||
| 43 | static int sh_csi2_try_fmt(struct v4l2_subdev *sd, | 45 | static int sh_csi2_try_fmt(struct v4l2_subdev *sd, |
| @@ -208,6 +210,7 @@ static int sh_csi2_notify(struct notifier_block *nb, | |||
| 208 | case BUS_NOTIFY_BOUND_DRIVER: | 210 | case BUS_NOTIFY_BOUND_DRIVER: |
| 209 | snprintf(priv->subdev.name, V4L2_SUBDEV_NAME_SIZE, "%s%s", | 211 | snprintf(priv->subdev.name, V4L2_SUBDEV_NAME_SIZE, "%s%s", |
| 210 | dev_name(v4l2_dev->dev), ".mipi-csi"); | 212 | dev_name(v4l2_dev->dev), ".mipi-csi"); |
| 213 | priv->subdev.grp_id = (long)icd; | ||
| 211 | ret = v4l2_device_register_subdev(v4l2_dev, &priv->subdev); | 214 | ret = v4l2_device_register_subdev(v4l2_dev, &priv->subdev); |
| 212 | dev_dbg(dev, "%s(%p): ret(register_subdev) = %d\n", __func__, priv, ret); | 215 | dev_dbg(dev, "%s(%p): ret(register_subdev) = %d\n", __func__, priv, ret); |
| 213 | if (ret < 0) | 216 | if (ret < 0) |
| @@ -215,6 +218,8 @@ static int sh_csi2_notify(struct notifier_block *nb, | |||
| 215 | 218 | ||
| 216 | priv->client = pdata->clients + i; | 219 | priv->client = pdata->clients + i; |
| 217 | 220 | ||
| 221 | priv->set_bus_param = icd->ops->set_bus_param; | ||
| 222 | priv->query_bus_param = icd->ops->query_bus_param; | ||
| 218 | icd->ops->set_bus_param = sh_csi2_set_bus_param; | 223 | icd->ops->set_bus_param = sh_csi2_set_bus_param; |
| 219 | icd->ops->query_bus_param = sh_csi2_query_bus_param; | 224 | icd->ops->query_bus_param = sh_csi2_query_bus_param; |
| 220 | 225 | ||
| @@ -226,8 +231,10 @@ static int sh_csi2_notify(struct notifier_block *nb, | |||
| 226 | priv->client = NULL; | 231 | priv->client = NULL; |
| 227 | 232 | ||
| 228 | /* Driver is about to be unbound */ | 233 | /* Driver is about to be unbound */ |
| 229 | icd->ops->set_bus_param = NULL; | 234 | icd->ops->set_bus_param = priv->set_bus_param; |
| 230 | icd->ops->query_bus_param = NULL; | 235 | icd->ops->query_bus_param = priv->query_bus_param; |
| 236 | priv->set_bus_param = NULL; | ||
| 237 | priv->query_bus_param = NULL; | ||
| 231 | 238 | ||
| 232 | v4l2_device_unregister_subdev(&priv->subdev); | 239 | v4l2_device_unregister_subdev(&priv->subdev); |
| 233 | 240 | ||
diff --git a/drivers/media/video/soc_camera.c b/drivers/media/video/soc_camera.c index 46284489e4eb..3973f9a94753 100644 --- a/drivers/media/video/soc_camera.c +++ b/drivers/media/video/soc_camera.c | |||
| @@ -996,10 +996,11 @@ static void soc_camera_free_i2c(struct soc_camera_device *icd) | |||
| 996 | { | 996 | { |
| 997 | struct i2c_client *client = | 997 | struct i2c_client *client = |
| 998 | to_i2c_client(to_soc_camera_control(icd)); | 998 | to_i2c_client(to_soc_camera_control(icd)); |
| 999 | struct i2c_adapter *adap = client->adapter; | ||
| 999 | dev_set_drvdata(&icd->dev, NULL); | 1000 | dev_set_drvdata(&icd->dev, NULL); |
| 1000 | v4l2_device_unregister_subdev(i2c_get_clientdata(client)); | 1001 | v4l2_device_unregister_subdev(i2c_get_clientdata(client)); |
| 1001 | i2c_unregister_device(client); | 1002 | i2c_unregister_device(client); |
| 1002 | i2c_put_adapter(client->adapter); | 1003 | i2c_put_adapter(adap); |
| 1003 | } | 1004 | } |
| 1004 | #else | 1005 | #else |
| 1005 | #define soc_camera_init_i2c(icd, icl) (-ENODEV) | 1006 | #define soc_camera_init_i2c(icd, icl) (-ENODEV) |
| @@ -1071,6 +1072,9 @@ static int soc_camera_probe(struct device *dev) | |||
| 1071 | } | 1072 | } |
| 1072 | } | 1073 | } |
| 1073 | 1074 | ||
| 1075 | sd = soc_camera_to_subdev(icd); | ||
| 1076 | sd->grp_id = (long)icd; | ||
| 1077 | |||
| 1074 | /* At this point client .probe() should have run already */ | 1078 | /* At this point client .probe() should have run already */ |
| 1075 | ret = soc_camera_init_user_formats(icd); | 1079 | ret = soc_camera_init_user_formats(icd); |
| 1076 | if (ret < 0) | 1080 | if (ret < 0) |
| @@ -1092,7 +1096,6 @@ static int soc_camera_probe(struct device *dev) | |||
| 1092 | goto evidstart; | 1096 | goto evidstart; |
| 1093 | 1097 | ||
| 1094 | /* Try to improve our guess of a reasonable window format */ | 1098 | /* Try to improve our guess of a reasonable window format */ |
| 1095 | sd = soc_camera_to_subdev(icd); | ||
| 1096 | if (!v4l2_subdev_call(sd, video, g_mbus_fmt, &mf)) { | 1099 | if (!v4l2_subdev_call(sd, video, g_mbus_fmt, &mf)) { |
| 1097 | icd->user_width = mf.width; | 1100 | icd->user_width = mf.width; |
| 1098 | icd->user_height = mf.height; | 1101 | icd->user_height = mf.height; |
diff --git a/drivers/media/video/v4l2-dev.c b/drivers/media/video/v4l2-dev.c index 498e6742579e..6dc7196296b3 100644 --- a/drivers/media/video/v4l2-dev.c +++ b/drivers/media/video/v4l2-dev.c | |||
| @@ -389,7 +389,8 @@ static int v4l2_open(struct inode *inode, struct file *filp) | |||
| 389 | video_get(vdev); | 389 | video_get(vdev); |
| 390 | mutex_unlock(&videodev_lock); | 390 | mutex_unlock(&videodev_lock); |
| 391 | #if defined(CONFIG_MEDIA_CONTROLLER) | 391 | #if defined(CONFIG_MEDIA_CONTROLLER) |
| 392 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) { | 392 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev && |
| 393 | vdev->vfl_type != VFL_TYPE_SUBDEV) { | ||
| 393 | entity = media_entity_get(&vdev->entity); | 394 | entity = media_entity_get(&vdev->entity); |
| 394 | if (!entity) { | 395 | if (!entity) { |
| 395 | ret = -EBUSY; | 396 | ret = -EBUSY; |
| @@ -415,7 +416,8 @@ err: | |||
| 415 | /* decrease the refcount in case of an error */ | 416 | /* decrease the refcount in case of an error */ |
| 416 | if (ret) { | 417 | if (ret) { |
| 417 | #if defined(CONFIG_MEDIA_CONTROLLER) | 418 | #if defined(CONFIG_MEDIA_CONTROLLER) |
| 418 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) | 419 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev && |
| 420 | vdev->vfl_type != VFL_TYPE_SUBDEV) | ||
| 419 | media_entity_put(entity); | 421 | media_entity_put(entity); |
| 420 | #endif | 422 | #endif |
| 421 | video_put(vdev); | 423 | video_put(vdev); |
| @@ -437,7 +439,8 @@ static int v4l2_release(struct inode *inode, struct file *filp) | |||
| 437 | mutex_unlock(vdev->lock); | 439 | mutex_unlock(vdev->lock); |
| 438 | } | 440 | } |
| 439 | #if defined(CONFIG_MEDIA_CONTROLLER) | 441 | #if defined(CONFIG_MEDIA_CONTROLLER) |
| 440 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) | 442 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev && |
| 443 | vdev->vfl_type != VFL_TYPE_SUBDEV) | ||
| 441 | media_entity_put(&vdev->entity); | 444 | media_entity_put(&vdev->entity); |
| 442 | #endif | 445 | #endif |
| 443 | /* decrease the refcount unconditionally since the release() | 446 | /* decrease the refcount unconditionally since the release() |
| @@ -686,7 +689,8 @@ int __video_register_device(struct video_device *vdev, int type, int nr, | |||
| 686 | 689 | ||
| 687 | #if defined(CONFIG_MEDIA_CONTROLLER) | 690 | #if defined(CONFIG_MEDIA_CONTROLLER) |
| 688 | /* Part 5: Register the entity. */ | 691 | /* Part 5: Register the entity. */ |
| 689 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) { | 692 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev && |
| 693 | vdev->vfl_type != VFL_TYPE_SUBDEV) { | ||
| 690 | vdev->entity.type = MEDIA_ENT_T_DEVNODE_V4L; | 694 | vdev->entity.type = MEDIA_ENT_T_DEVNODE_V4L; |
| 691 | vdev->entity.name = vdev->name; | 695 | vdev->entity.name = vdev->name; |
| 692 | vdev->entity.v4l.major = VIDEO_MAJOR; | 696 | vdev->entity.v4l.major = VIDEO_MAJOR; |
| @@ -733,7 +737,8 @@ void video_unregister_device(struct video_device *vdev) | |||
| 733 | return; | 737 | return; |
| 734 | 738 | ||
| 735 | #if defined(CONFIG_MEDIA_CONTROLLER) | 739 | #if defined(CONFIG_MEDIA_CONTROLLER) |
| 736 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev) | 740 | if (vdev->v4l2_dev && vdev->v4l2_dev->mdev && |
| 741 | vdev->vfl_type != VFL_TYPE_SUBDEV) | ||
| 737 | media_device_unregister_entity(&vdev->entity); | 742 | media_device_unregister_entity(&vdev->entity); |
| 738 | #endif | 743 | #endif |
| 739 | 744 | ||
diff --git a/drivers/media/video/videobuf2-core.c b/drivers/media/video/videobuf2-core.c index 6698c77e0f64..6ba1461d51ef 100644 --- a/drivers/media/video/videobuf2-core.c +++ b/drivers/media/video/videobuf2-core.c | |||
| @@ -37,6 +37,9 @@ module_param(debug, int, 0644); | |||
| 37 | #define call_qop(q, op, args...) \ | 37 | #define call_qop(q, op, args...) \ |
| 38 | (((q)->ops->op) ? ((q)->ops->op(args)) : 0) | 38 | (((q)->ops->op) ? ((q)->ops->op(args)) : 0) |
| 39 | 39 | ||
| 40 | #define V4L2_BUFFER_STATE_FLAGS (V4L2_BUF_FLAG_MAPPED | V4L2_BUF_FLAG_QUEUED | \ | ||
| 41 | V4L2_BUF_FLAG_DONE | V4L2_BUF_FLAG_ERROR) | ||
| 42 | |||
| 40 | /** | 43 | /** |
| 41 | * __vb2_buf_mem_alloc() - allocate video memory for the given buffer | 44 | * __vb2_buf_mem_alloc() - allocate video memory for the given buffer |
| 42 | */ | 45 | */ |
| @@ -51,7 +54,7 @@ static int __vb2_buf_mem_alloc(struct vb2_buffer *vb, | |||
| 51 | for (plane = 0; plane < vb->num_planes; ++plane) { | 54 | for (plane = 0; plane < vb->num_planes; ++plane) { |
| 52 | mem_priv = call_memop(q, plane, alloc, q->alloc_ctx[plane], | 55 | mem_priv = call_memop(q, plane, alloc, q->alloc_ctx[plane], |
| 53 | plane_sizes[plane]); | 56 | plane_sizes[plane]); |
| 54 | if (!mem_priv) | 57 | if (IS_ERR_OR_NULL(mem_priv)) |
| 55 | goto free; | 58 | goto free; |
| 56 | 59 | ||
| 57 | /* Associate allocator private data with this plane */ | 60 | /* Associate allocator private data with this plane */ |
| @@ -284,7 +287,7 @@ static int __fill_v4l2_buffer(struct vb2_buffer *vb, struct v4l2_buffer *b) | |||
| 284 | struct vb2_queue *q = vb->vb2_queue; | 287 | struct vb2_queue *q = vb->vb2_queue; |
| 285 | int ret = 0; | 288 | int ret = 0; |
| 286 | 289 | ||
| 287 | /* Copy back data such as timestamp, input, etc. */ | 290 | /* Copy back data such as timestamp, flags, input, etc. */ |
| 288 | memcpy(b, &vb->v4l2_buf, offsetof(struct v4l2_buffer, m)); | 291 | memcpy(b, &vb->v4l2_buf, offsetof(struct v4l2_buffer, m)); |
| 289 | b->input = vb->v4l2_buf.input; | 292 | b->input = vb->v4l2_buf.input; |
| 290 | b->reserved = vb->v4l2_buf.reserved; | 293 | b->reserved = vb->v4l2_buf.reserved; |
| @@ -313,7 +316,10 @@ static int __fill_v4l2_buffer(struct vb2_buffer *vb, struct v4l2_buffer *b) | |||
| 313 | b->m.userptr = vb->v4l2_planes[0].m.userptr; | 316 | b->m.userptr = vb->v4l2_planes[0].m.userptr; |
| 314 | } | 317 | } |
| 315 | 318 | ||
| 316 | b->flags = 0; | 319 | /* |
| 320 | * Clear any buffer state related flags. | ||
| 321 | */ | ||
| 322 | b->flags &= ~V4L2_BUFFER_STATE_FLAGS; | ||
| 317 | 323 | ||
| 318 | switch (vb->state) { | 324 | switch (vb->state) { |
| 319 | case VB2_BUF_STATE_QUEUED: | 325 | case VB2_BUF_STATE_QUEUED: |
| @@ -519,6 +525,7 @@ int vb2_reqbufs(struct vb2_queue *q, struct v4l2_requestbuffers *req) | |||
| 519 | num_buffers = min_t(unsigned int, req->count, VIDEO_MAX_FRAME); | 525 | num_buffers = min_t(unsigned int, req->count, VIDEO_MAX_FRAME); |
| 520 | memset(plane_sizes, 0, sizeof(plane_sizes)); | 526 | memset(plane_sizes, 0, sizeof(plane_sizes)); |
| 521 | memset(q->alloc_ctx, 0, sizeof(q->alloc_ctx)); | 527 | memset(q->alloc_ctx, 0, sizeof(q->alloc_ctx)); |
| 528 | q->memory = req->memory; | ||
| 522 | 529 | ||
| 523 | /* | 530 | /* |
| 524 | * Ask the driver how many buffers and planes per buffer it requires. | 531 | * Ask the driver how many buffers and planes per buffer it requires. |
| @@ -560,8 +567,6 @@ int vb2_reqbufs(struct vb2_queue *q, struct v4l2_requestbuffers *req) | |||
| 560 | ret = num_buffers; | 567 | ret = num_buffers; |
| 561 | } | 568 | } |
| 562 | 569 | ||
| 563 | q->memory = req->memory; | ||
| 564 | |||
| 565 | /* | 570 | /* |
| 566 | * Return the number of successfully allocated buffers | 571 | * Return the number of successfully allocated buffers |
| 567 | * to the userspace. | 572 | * to the userspace. |
| @@ -715,6 +720,8 @@ static int __fill_vb2_buffer(struct vb2_buffer *vb, struct v4l2_buffer *b, | |||
| 715 | 720 | ||
| 716 | vb->v4l2_buf.field = b->field; | 721 | vb->v4l2_buf.field = b->field; |
| 717 | vb->v4l2_buf.timestamp = b->timestamp; | 722 | vb->v4l2_buf.timestamp = b->timestamp; |
| 723 | vb->v4l2_buf.input = b->input; | ||
| 724 | vb->v4l2_buf.flags = b->flags & ~V4L2_BUFFER_STATE_FLAGS; | ||
| 718 | 725 | ||
| 719 | return 0; | 726 | return 0; |
| 720 | } | 727 | } |
diff --git a/drivers/media/video/videobuf2-dma-contig.c b/drivers/media/video/videobuf2-dma-contig.c index 58205d596138..a790a5f8c06f 100644 --- a/drivers/media/video/videobuf2-dma-contig.c +++ b/drivers/media/video/videobuf2-dma-contig.c | |||
| @@ -46,7 +46,7 @@ static void *vb2_dma_contig_alloc(void *alloc_ctx, unsigned long size) | |||
| 46 | GFP_KERNEL); | 46 | GFP_KERNEL); |
| 47 | if (!buf->vaddr) { | 47 | if (!buf->vaddr) { |
| 48 | dev_err(conf->dev, "dma_alloc_coherent of size %ld failed\n", | 48 | dev_err(conf->dev, "dma_alloc_coherent of size %ld failed\n", |
| 49 | buf->size); | 49 | size); |
| 50 | kfree(buf); | 50 | kfree(buf); |
| 51 | return ERR_PTR(-ENOMEM); | 51 | return ERR_PTR(-ENOMEM); |
| 52 | } | 52 | } |
diff --git a/drivers/mmc/core/bus.c b/drivers/mmc/core/bus.c index 63667a8f140c..d6d62fd07ee9 100644 --- a/drivers/mmc/core/bus.c +++ b/drivers/mmc/core/bus.c | |||
| @@ -284,6 +284,7 @@ int mmc_add_card(struct mmc_card *card) | |||
| 284 | type = "SD-combo"; | 284 | type = "SD-combo"; |
| 285 | if (mmc_card_blockaddr(card)) | 285 | if (mmc_card_blockaddr(card)) |
| 286 | type = "SDHC-combo"; | 286 | type = "SDHC-combo"; |
| 287 | break; | ||
| 287 | default: | 288 | default: |
| 288 | type = "?"; | 289 | type = "?"; |
| 289 | break; | 290 | break; |
diff --git a/drivers/mmc/core/host.c b/drivers/mmc/core/host.c index 461e6a17fb90..2b200c1cfbba 100644 --- a/drivers/mmc/core/host.c +++ b/drivers/mmc/core/host.c | |||
| @@ -94,7 +94,7 @@ static void mmc_host_clk_gate_delayed(struct mmc_host *host) | |||
| 94 | spin_unlock_irqrestore(&host->clk_lock, flags); | 94 | spin_unlock_irqrestore(&host->clk_lock, flags); |
| 95 | return; | 95 | return; |
| 96 | } | 96 | } |
| 97 | mutex_lock(&host->clk_gate_mutex); | 97 | mmc_claim_host(host); |
| 98 | spin_lock_irqsave(&host->clk_lock, flags); | 98 | spin_lock_irqsave(&host->clk_lock, flags); |
| 99 | if (!host->clk_requests) { | 99 | if (!host->clk_requests) { |
| 100 | spin_unlock_irqrestore(&host->clk_lock, flags); | 100 | spin_unlock_irqrestore(&host->clk_lock, flags); |
| @@ -104,7 +104,7 @@ static void mmc_host_clk_gate_delayed(struct mmc_host *host) | |||
| 104 | pr_debug("%s: gated MCI clock\n", mmc_hostname(host)); | 104 | pr_debug("%s: gated MCI clock\n", mmc_hostname(host)); |
| 105 | } | 105 | } |
| 106 | spin_unlock_irqrestore(&host->clk_lock, flags); | 106 | spin_unlock_irqrestore(&host->clk_lock, flags); |
| 107 | mutex_unlock(&host->clk_gate_mutex); | 107 | mmc_release_host(host); |
| 108 | } | 108 | } |
| 109 | 109 | ||
| 110 | /* | 110 | /* |
| @@ -130,7 +130,7 @@ void mmc_host_clk_ungate(struct mmc_host *host) | |||
| 130 | { | 130 | { |
| 131 | unsigned long flags; | 131 | unsigned long flags; |
| 132 | 132 | ||
| 133 | mutex_lock(&host->clk_gate_mutex); | 133 | mmc_claim_host(host); |
| 134 | spin_lock_irqsave(&host->clk_lock, flags); | 134 | spin_lock_irqsave(&host->clk_lock, flags); |
| 135 | if (host->clk_gated) { | 135 | if (host->clk_gated) { |
| 136 | spin_unlock_irqrestore(&host->clk_lock, flags); | 136 | spin_unlock_irqrestore(&host->clk_lock, flags); |
| @@ -140,7 +140,7 @@ void mmc_host_clk_ungate(struct mmc_host *host) | |||
| 140 | } | 140 | } |
| 141 | host->clk_requests++; | 141 | host->clk_requests++; |
| 142 | spin_unlock_irqrestore(&host->clk_lock, flags); | 142 | spin_unlock_irqrestore(&host->clk_lock, flags); |
| 143 | mutex_unlock(&host->clk_gate_mutex); | 143 | mmc_release_host(host); |
| 144 | } | 144 | } |
| 145 | 145 | ||
| 146 | /** | 146 | /** |
| @@ -215,7 +215,6 @@ static inline void mmc_host_clk_init(struct mmc_host *host) | |||
| 215 | host->clk_gated = false; | 215 | host->clk_gated = false; |
| 216 | INIT_WORK(&host->clk_gate_work, mmc_host_clk_gate_work); | 216 | INIT_WORK(&host->clk_gate_work, mmc_host_clk_gate_work); |
| 217 | spin_lock_init(&host->clk_lock); | 217 | spin_lock_init(&host->clk_lock); |
| 218 | mutex_init(&host->clk_gate_mutex); | ||
| 219 | } | 218 | } |
| 220 | 219 | ||
| 221 | /** | 220 | /** |
diff --git a/drivers/mmc/host/omap.c b/drivers/mmc/host/omap.c index 2e032f0e8cf4..a6c329040140 100644 --- a/drivers/mmc/host/omap.c +++ b/drivers/mmc/host/omap.c | |||
| @@ -832,7 +832,7 @@ static irqreturn_t mmc_omap_irq(int irq, void *dev_id) | |||
| 832 | return IRQ_HANDLED; | 832 | return IRQ_HANDLED; |
| 833 | } | 833 | } |
| 834 | 834 | ||
| 835 | if (end_command) | 835 | if (end_command && host->cmd) |
| 836 | mmc_omap_cmd_done(host, host->cmd); | 836 | mmc_omap_cmd_done(host, host->cmd); |
| 837 | if (host->data != NULL) { | 837 | if (host->data != NULL) { |
| 838 | if (transfer_error) | 838 | if (transfer_error) |
diff --git a/drivers/mmc/host/sdhci-pci.c b/drivers/mmc/host/sdhci-pci.c index a136be706347..f8b5f37007b2 100644 --- a/drivers/mmc/host/sdhci-pci.c +++ b/drivers/mmc/host/sdhci-pci.c | |||
| @@ -957,6 +957,7 @@ static struct sdhci_pci_slot * __devinit sdhci_pci_probe_slot( | |||
| 957 | host->ioaddr = pci_ioremap_bar(pdev, bar); | 957 | host->ioaddr = pci_ioremap_bar(pdev, bar); |
| 958 | if (!host->ioaddr) { | 958 | if (!host->ioaddr) { |
| 959 | dev_err(&pdev->dev, "failed to remap registers\n"); | 959 | dev_err(&pdev->dev, "failed to remap registers\n"); |
| 960 | ret = -ENOMEM; | ||
| 960 | goto release; | 961 | goto release; |
| 961 | } | 962 | } |
| 962 | 963 | ||
diff --git a/drivers/mmc/host/sdhci.c b/drivers/mmc/host/sdhci.c index 9e15f41f87be..5d20661bc357 100644 --- a/drivers/mmc/host/sdhci.c +++ b/drivers/mmc/host/sdhci.c | |||
| @@ -1334,6 +1334,13 @@ static void sdhci_tasklet_finish(unsigned long param) | |||
| 1334 | 1334 | ||
| 1335 | host = (struct sdhci_host*)param; | 1335 | host = (struct sdhci_host*)param; |
| 1336 | 1336 | ||
| 1337 | /* | ||
| 1338 | * If this tasklet gets rescheduled while running, it will | ||
| 1339 | * be run again afterwards but without any active request. | ||
| 1340 | */ | ||
| 1341 | if (!host->mrq) | ||
| 1342 | return; | ||
| 1343 | |||
| 1337 | spin_lock_irqsave(&host->lock, flags); | 1344 | spin_lock_irqsave(&host->lock, flags); |
| 1338 | 1345 | ||
| 1339 | del_timer(&host->timer); | 1346 | del_timer(&host->timer); |
| @@ -1345,7 +1352,7 @@ static void sdhci_tasklet_finish(unsigned long param) | |||
| 1345 | * upon error conditions. | 1352 | * upon error conditions. |
| 1346 | */ | 1353 | */ |
| 1347 | if (!(host->flags & SDHCI_DEVICE_DEAD) && | 1354 | if (!(host->flags & SDHCI_DEVICE_DEAD) && |
| 1348 | (mrq->cmd->error || | 1355 | ((mrq->cmd && mrq->cmd->error) || |
| 1349 | (mrq->data && (mrq->data->error || | 1356 | (mrq->data && (mrq->data->error || |
| 1350 | (mrq->data->stop && mrq->data->stop->error))) || | 1357 | (mrq->data->stop && mrq->data->stop->error))) || |
| 1351 | (host->quirks & SDHCI_QUIRK_RESET_AFTER_REQUEST))) { | 1358 | (host->quirks & SDHCI_QUIRK_RESET_AFTER_REQUEST))) { |
diff --git a/drivers/mmc/host/tmio_mmc_pio.c b/drivers/mmc/host/tmio_mmc_pio.c index 62d37de6de76..710339a85c84 100644 --- a/drivers/mmc/host/tmio_mmc_pio.c +++ b/drivers/mmc/host/tmio_mmc_pio.c | |||
| @@ -728,15 +728,15 @@ static void tmio_mmc_set_ios(struct mmc_host *mmc, struct mmc_ios *ios) | |||
| 728 | tmio_mmc_set_clock(host, ios->clock); | 728 | tmio_mmc_set_clock(host, ios->clock); |
| 729 | 729 | ||
| 730 | /* Power sequence - OFF -> UP -> ON */ | 730 | /* Power sequence - OFF -> UP -> ON */ |
| 731 | if (ios->power_mode == MMC_POWER_OFF || !ios->clock) { | 731 | if (ios->power_mode == MMC_POWER_UP) { |
| 732 | /* power up SD bus */ | ||
| 733 | if (host->set_pwr) | ||
| 734 | host->set_pwr(host->pdev, 1); | ||
| 735 | } else if (ios->power_mode == MMC_POWER_OFF || !ios->clock) { | ||
| 732 | /* power down SD bus */ | 736 | /* power down SD bus */ |
| 733 | if (ios->power_mode == MMC_POWER_OFF && host->set_pwr) | 737 | if (ios->power_mode == MMC_POWER_OFF && host->set_pwr) |
| 734 | host->set_pwr(host->pdev, 0); | 738 | host->set_pwr(host->pdev, 0); |
| 735 | tmio_mmc_clk_stop(host); | 739 | tmio_mmc_clk_stop(host); |
| 736 | } else if (ios->power_mode == MMC_POWER_UP) { | ||
| 737 | /* power up SD bus */ | ||
| 738 | if (host->set_pwr) | ||
| 739 | host->set_pwr(host->pdev, 1); | ||
| 740 | } else { | 740 | } else { |
| 741 | /* start bus clock */ | 741 | /* start bus clock */ |
| 742 | tmio_mmc_clk_start(host); | 742 | tmio_mmc_clk_start(host); |
diff --git a/drivers/mtd/nand/diskonchip.c b/drivers/mtd/nand/diskonchip.c index 96c0b34ba8db..657b9f4b6f9b 100644 --- a/drivers/mtd/nand/diskonchip.c +++ b/drivers/mtd/nand/diskonchip.c | |||
| @@ -400,7 +400,7 @@ static uint16_t __init doc200x_ident_chip(struct mtd_info *mtd, int nr) | |||
| 400 | doc200x_hwcontrol(mtd, 0, NAND_CTRL_ALE | NAND_CTRL_CHANGE); | 400 | doc200x_hwcontrol(mtd, 0, NAND_CTRL_ALE | NAND_CTRL_CHANGE); |
| 401 | doc200x_hwcontrol(mtd, NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE); | 401 | doc200x_hwcontrol(mtd, NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE); |
| 402 | 402 | ||
| 403 | /* We can't' use dev_ready here, but at least we wait for the | 403 | /* We can't use dev_ready here, but at least we wait for the |
| 404 | * command to complete | 404 | * command to complete |
| 405 | */ | 405 | */ |
| 406 | udelay(50); | 406 | udelay(50); |
diff --git a/drivers/net/amd8111e.c b/drivers/net/amd8111e.c index 88495c48a81d..241b185e6569 100644 --- a/drivers/net/amd8111e.c +++ b/drivers/net/amd8111e.c | |||
| @@ -106,7 +106,7 @@ MODULE_DESCRIPTION ("AMD8111 based 10/100 Ethernet Controller. Driver Version "M | |||
| 106 | MODULE_LICENSE("GPL"); | 106 | MODULE_LICENSE("GPL"); |
| 107 | MODULE_DEVICE_TABLE(pci, amd8111e_pci_tbl); | 107 | MODULE_DEVICE_TABLE(pci, amd8111e_pci_tbl); |
| 108 | module_param_array(speed_duplex, int, NULL, 0); | 108 | module_param_array(speed_duplex, int, NULL, 0); |
| 109 | MODULE_PARM_DESC(speed_duplex, "Set device speed and duplex modes, 0: Auto Negotitate, 1: 10Mbps Half Duplex, 2: 10Mbps Full Duplex, 3: 100Mbps Half Duplex, 4: 100Mbps Full Duplex"); | 109 | MODULE_PARM_DESC(speed_duplex, "Set device speed and duplex modes, 0: Auto Negotiate, 1: 10Mbps Half Duplex, 2: 10Mbps Full Duplex, 3: 100Mbps Half Duplex, 4: 100Mbps Full Duplex"); |
| 110 | module_param_array(coalesce, bool, NULL, 0); | 110 | module_param_array(coalesce, bool, NULL, 0); |
| 111 | MODULE_PARM_DESC(coalesce, "Enable or Disable interrupt coalescing, 1: Enable, 0: Disable"); | 111 | MODULE_PARM_DESC(coalesce, "Enable or Disable interrupt coalescing, 1: Enable, 0: Disable"); |
| 112 | module_param_array(dynamic_ipg, bool, NULL, 0); | 112 | module_param_array(dynamic_ipg, bool, NULL, 0); |
diff --git a/drivers/net/atl1c/atl1c.h b/drivers/net/atl1c/atl1c.h index 7cb375e0e29c..925929d764ca 100644 --- a/drivers/net/atl1c/atl1c.h +++ b/drivers/net/atl1c/atl1c.h | |||
| @@ -566,9 +566,9 @@ struct atl1c_adapter { | |||
| 566 | #define __AT_TESTING 0x0001 | 566 | #define __AT_TESTING 0x0001 |
| 567 | #define __AT_RESETTING 0x0002 | 567 | #define __AT_RESETTING 0x0002 |
| 568 | #define __AT_DOWN 0x0003 | 568 | #define __AT_DOWN 0x0003 |
| 569 | u8 work_event; | 569 | unsigned long work_event; |
| 570 | #define ATL1C_WORK_EVENT_RESET 0x01 | 570 | #define ATL1C_WORK_EVENT_RESET 0 |
| 571 | #define ATL1C_WORK_EVENT_LINK_CHANGE 0x02 | 571 | #define ATL1C_WORK_EVENT_LINK_CHANGE 1 |
| 572 | u32 msg_enable; | 572 | u32 msg_enable; |
| 573 | 573 | ||
| 574 | bool have_msi; | 574 | bool have_msi; |
diff --git a/drivers/net/atl1c/atl1c_main.c b/drivers/net/atl1c/atl1c_main.c index 7d9d5067a65c..a6e1c36e48e6 100644 --- a/drivers/net/atl1c/atl1c_main.c +++ b/drivers/net/atl1c/atl1c_main.c | |||
| @@ -325,7 +325,7 @@ static void atl1c_link_chg_event(struct atl1c_adapter *adapter) | |||
| 325 | } | 325 | } |
| 326 | } | 326 | } |
| 327 | 327 | ||
| 328 | adapter->work_event |= ATL1C_WORK_EVENT_LINK_CHANGE; | 328 | set_bit(ATL1C_WORK_EVENT_LINK_CHANGE, &adapter->work_event); |
| 329 | schedule_work(&adapter->common_task); | 329 | schedule_work(&adapter->common_task); |
| 330 | } | 330 | } |
| 331 | 331 | ||
| @@ -337,20 +337,16 @@ static void atl1c_common_task(struct work_struct *work) | |||
| 337 | adapter = container_of(work, struct atl1c_adapter, common_task); | 337 | adapter = container_of(work, struct atl1c_adapter, common_task); |
| 338 | netdev = adapter->netdev; | 338 | netdev = adapter->netdev; |
| 339 | 339 | ||
| 340 | if (adapter->work_event & ATL1C_WORK_EVENT_RESET) { | 340 | if (test_and_clear_bit(ATL1C_WORK_EVENT_RESET, &adapter->work_event)) { |
| 341 | adapter->work_event &= ~ATL1C_WORK_EVENT_RESET; | ||
| 342 | netif_device_detach(netdev); | 341 | netif_device_detach(netdev); |
| 343 | atl1c_down(adapter); | 342 | atl1c_down(adapter); |
| 344 | atl1c_up(adapter); | 343 | atl1c_up(adapter); |
| 345 | netif_device_attach(netdev); | 344 | netif_device_attach(netdev); |
| 346 | return; | ||
| 347 | } | 345 | } |
| 348 | 346 | ||
| 349 | if (adapter->work_event & ATL1C_WORK_EVENT_LINK_CHANGE) { | 347 | if (test_and_clear_bit(ATL1C_WORK_EVENT_LINK_CHANGE, |
| 350 | adapter->work_event &= ~ATL1C_WORK_EVENT_LINK_CHANGE; | 348 | &adapter->work_event)) |
| 351 | atl1c_check_link_status(adapter); | 349 | atl1c_check_link_status(adapter); |
| 352 | } | ||
| 353 | return; | ||
| 354 | } | 350 | } |
| 355 | 351 | ||
| 356 | 352 | ||
| @@ -369,7 +365,7 @@ static void atl1c_tx_timeout(struct net_device *netdev) | |||
| 369 | struct atl1c_adapter *adapter = netdev_priv(netdev); | 365 | struct atl1c_adapter *adapter = netdev_priv(netdev); |
| 370 | 366 | ||
| 371 | /* Do the reset outside of interrupt context */ | 367 | /* Do the reset outside of interrupt context */ |
| 372 | adapter->work_event |= ATL1C_WORK_EVENT_RESET; | 368 | set_bit(ATL1C_WORK_EVENT_RESET, &adapter->work_event); |
| 373 | schedule_work(&adapter->common_task); | 369 | schedule_work(&adapter->common_task); |
| 374 | } | 370 | } |
| 375 | 371 | ||
diff --git a/drivers/net/benet/be_main.c b/drivers/net/benet/be_main.c index 7cb5a114c733..02a0443d1821 100644 --- a/drivers/net/benet/be_main.c +++ b/drivers/net/benet/be_main.c | |||
| @@ -1873,6 +1873,7 @@ static void be_worker(struct work_struct *work) | |||
| 1873 | be_detect_dump_ue(adapter); | 1873 | be_detect_dump_ue(adapter); |
| 1874 | 1874 | ||
| 1875 | reschedule: | 1875 | reschedule: |
| 1876 | adapter->work_counter++; | ||
| 1876 | schedule_delayed_work(&adapter->work, msecs_to_jiffies(1000)); | 1877 | schedule_delayed_work(&adapter->work, msecs_to_jiffies(1000)); |
| 1877 | } | 1878 | } |
| 1878 | 1879 | ||
diff --git a/drivers/net/bnx2.c b/drivers/net/bnx2.c index 8e6d618b5305..d8383a9af9ad 100644 --- a/drivers/net/bnx2.c +++ b/drivers/net/bnx2.c | |||
| @@ -8413,6 +8413,8 @@ bnx2_remove_one(struct pci_dev *pdev) | |||
| 8413 | 8413 | ||
| 8414 | unregister_netdev(dev); | 8414 | unregister_netdev(dev); |
| 8415 | 8415 | ||
| 8416 | del_timer_sync(&bp->timer); | ||
| 8417 | |||
| 8416 | if (bp->mips_firmware) | 8418 | if (bp->mips_firmware) |
| 8417 | release_firmware(bp->mips_firmware); | 8419 | release_firmware(bp->mips_firmware); |
| 8418 | if (bp->rv2p_firmware) | 8420 | if (bp->rv2p_firmware) |
diff --git a/drivers/net/bnx2x/bnx2x_cmn.c b/drivers/net/bnx2x/bnx2x_cmn.c index e83ac6dd6fc0..16581df5ee4e 100644 --- a/drivers/net/bnx2x/bnx2x_cmn.c +++ b/drivers/net/bnx2x/bnx2x_cmn.c | |||
| @@ -2019,15 +2019,23 @@ static inline void bnx2x_set_pbd_gso(struct sk_buff *skb, | |||
| 2019 | static inline u8 bnx2x_set_pbd_csum_e2(struct bnx2x *bp, struct sk_buff *skb, | 2019 | static inline u8 bnx2x_set_pbd_csum_e2(struct bnx2x *bp, struct sk_buff *skb, |
| 2020 | u32 *parsing_data, u32 xmit_type) | 2020 | u32 *parsing_data, u32 xmit_type) |
| 2021 | { | 2021 | { |
| 2022 | *parsing_data |= ((tcp_hdrlen(skb)/4) << | 2022 | *parsing_data |= |
| 2023 | ETH_TX_PARSE_BD_E2_TCP_HDR_LENGTH_DW_SHIFT) & | 2023 | ((((u8 *)skb_transport_header(skb) - skb->data) >> 1) << |
| 2024 | ETH_TX_PARSE_BD_E2_TCP_HDR_LENGTH_DW; | 2024 | ETH_TX_PARSE_BD_E2_TCP_HDR_START_OFFSET_W_SHIFT) & |
| 2025 | ETH_TX_PARSE_BD_E2_TCP_HDR_START_OFFSET_W; | ||
| 2025 | 2026 | ||
| 2026 | *parsing_data |= ((((u8 *)tcp_hdr(skb) - skb->data) / 2) << | 2027 | if (xmit_type & XMIT_CSUM_TCP) { |
| 2027 | ETH_TX_PARSE_BD_E2_TCP_HDR_START_OFFSET_W_SHIFT) & | 2028 | *parsing_data |= ((tcp_hdrlen(skb) / 4) << |
| 2028 | ETH_TX_PARSE_BD_E2_TCP_HDR_START_OFFSET_W; | 2029 | ETH_TX_PARSE_BD_E2_TCP_HDR_LENGTH_DW_SHIFT) & |
| 2030 | ETH_TX_PARSE_BD_E2_TCP_HDR_LENGTH_DW; | ||
| 2029 | 2031 | ||
| 2030 | return skb_transport_header(skb) + tcp_hdrlen(skb) - skb->data; | 2032 | return skb_transport_header(skb) + tcp_hdrlen(skb) - skb->data; |
| 2033 | } else | ||
| 2034 | /* We support checksum offload for TCP and UDP only. | ||
| 2035 | * No need to pass the UDP header length - it's a constant. | ||
| 2036 | */ | ||
| 2037 | return skb_transport_header(skb) + | ||
| 2038 | sizeof(struct udphdr) - skb->data; | ||
| 2031 | } | 2039 | } |
| 2032 | 2040 | ||
| 2033 | /** | 2041 | /** |
| @@ -2043,7 +2051,7 @@ static inline u8 bnx2x_set_pbd_csum(struct bnx2x *bp, struct sk_buff *skb, | |||
| 2043 | struct eth_tx_parse_bd_e1x *pbd, | 2051 | struct eth_tx_parse_bd_e1x *pbd, |
| 2044 | u32 xmit_type) | 2052 | u32 xmit_type) |
| 2045 | { | 2053 | { |
| 2046 | u8 hlen = (skb_network_header(skb) - skb->data) / 2; | 2054 | u8 hlen = (skb_network_header(skb) - skb->data) >> 1; |
| 2047 | 2055 | ||
| 2048 | /* for now NS flag is not used in Linux */ | 2056 | /* for now NS flag is not used in Linux */ |
| 2049 | pbd->global_data = | 2057 | pbd->global_data = |
| @@ -2051,9 +2059,15 @@ static inline u8 bnx2x_set_pbd_csum(struct bnx2x *bp, struct sk_buff *skb, | |||
| 2051 | ETH_TX_PARSE_BD_E1X_LLC_SNAP_EN_SHIFT)); | 2059 | ETH_TX_PARSE_BD_E1X_LLC_SNAP_EN_SHIFT)); |
| 2052 | 2060 | ||
| 2053 | pbd->ip_hlen_w = (skb_transport_header(skb) - | 2061 | pbd->ip_hlen_w = (skb_transport_header(skb) - |
| 2054 | skb_network_header(skb)) / 2; | 2062 | skb_network_header(skb)) >> 1; |
| 2055 | 2063 | ||
| 2056 | hlen += pbd->ip_hlen_w + tcp_hdrlen(skb) / 2; | 2064 | hlen += pbd->ip_hlen_w; |
| 2065 | |||
| 2066 | /* We support checksum offload for TCP and UDP only */ | ||
| 2067 | if (xmit_type & XMIT_CSUM_TCP) | ||
| 2068 | hlen += tcp_hdrlen(skb) / 2; | ||
| 2069 | else | ||
| 2070 | hlen += sizeof(struct udphdr) / 2; | ||
| 2057 | 2071 | ||
| 2058 | pbd->total_hlen_w = cpu_to_le16(hlen); | 2072 | pbd->total_hlen_w = cpu_to_le16(hlen); |
| 2059 | hlen = hlen*2; | 2073 | hlen = hlen*2; |
diff --git a/drivers/net/bonding/bond_3ad.c b/drivers/net/bonding/bond_3ad.c index 494bf960442d..31912f17653f 100644 --- a/drivers/net/bonding/bond_3ad.c +++ b/drivers/net/bonding/bond_3ad.c | |||
| @@ -1482,8 +1482,11 @@ static struct aggregator *ad_agg_selection_test(struct aggregator *best, | |||
| 1482 | 1482 | ||
| 1483 | static int agg_device_up(const struct aggregator *agg) | 1483 | static int agg_device_up(const struct aggregator *agg) |
| 1484 | { | 1484 | { |
| 1485 | return (netif_running(agg->slave->dev) && | 1485 | struct port *port = agg->lag_ports; |
| 1486 | netif_carrier_ok(agg->slave->dev)); | 1486 | if (!port) |
| 1487 | return 0; | ||
| 1488 | return (netif_running(port->slave->dev) && | ||
| 1489 | netif_carrier_ok(port->slave->dev)); | ||
| 1487 | } | 1490 | } |
| 1488 | 1491 | ||
| 1489 | /** | 1492 | /** |
diff --git a/drivers/net/ehea/ehea_main.c b/drivers/net/ehea/ehea_main.c index f75d3144b8a5..53c0f04b1b23 100644 --- a/drivers/net/ehea/ehea_main.c +++ b/drivers/net/ehea/ehea_main.c | |||
| @@ -3040,11 +3040,14 @@ static void ehea_rereg_mrs(void) | |||
| 3040 | 3040 | ||
| 3041 | if (dev->flags & IFF_UP) { | 3041 | if (dev->flags & IFF_UP) { |
| 3042 | mutex_lock(&port->port_lock); | 3042 | mutex_lock(&port->port_lock); |
| 3043 | port_napi_enable(port); | ||
| 3044 | ret = ehea_restart_qps(dev); | 3043 | ret = ehea_restart_qps(dev); |
| 3045 | check_sqs(port); | 3044 | if (!ret) { |
| 3046 | if (!ret) | 3045 | check_sqs(port); |
| 3046 | port_napi_enable(port); | ||
| 3047 | netif_wake_queue(dev); | 3047 | netif_wake_queue(dev); |
| 3048 | } else { | ||
| 3049 | netdev_err(dev, "Unable to restart QPS\n"); | ||
| 3050 | } | ||
| 3048 | mutex_unlock(&port->port_lock); | 3051 | mutex_unlock(&port->port_lock); |
| 3049 | } | 3052 | } |
| 3050 | } | 3053 | } |
diff --git a/drivers/net/fs_enet/mac-fec.c b/drivers/net/fs_enet/mac-fec.c index 61035fc5599b..b9fbc83d64a7 100644 --- a/drivers/net/fs_enet/mac-fec.c +++ b/drivers/net/fs_enet/mac-fec.c | |||
| @@ -226,8 +226,8 @@ static void set_multicast_finish(struct net_device *dev) | |||
| 226 | } | 226 | } |
| 227 | 227 | ||
| 228 | FC(fecp, r_cntrl, FEC_RCNTRL_PROM); | 228 | FC(fecp, r_cntrl, FEC_RCNTRL_PROM); |
| 229 | FW(fecp, hash_table_high, fep->fec.hthi); | 229 | FW(fecp, grp_hash_table_high, fep->fec.hthi); |
| 230 | FW(fecp, hash_table_low, fep->fec.htlo); | 230 | FW(fecp, grp_hash_table_low, fep->fec.htlo); |
| 231 | } | 231 | } |
| 232 | 232 | ||
| 233 | static void set_multicast_list(struct net_device *dev) | 233 | static void set_multicast_list(struct net_device *dev) |
| @@ -273,8 +273,8 @@ static void restart(struct net_device *dev) | |||
| 273 | /* | 273 | /* |
| 274 | * Reset all multicast. | 274 | * Reset all multicast. |
| 275 | */ | 275 | */ |
| 276 | FW(fecp, hash_table_high, fep->fec.hthi); | 276 | FW(fecp, grp_hash_table_high, fep->fec.hthi); |
| 277 | FW(fecp, hash_table_low, fep->fec.htlo); | 277 | FW(fecp, grp_hash_table_low, fep->fec.htlo); |
| 278 | 278 | ||
| 279 | /* | 279 | /* |
| 280 | * Set maximum receive buffer size. | 280 | * Set maximum receive buffer size. |
diff --git a/drivers/net/ftmac100.c b/drivers/net/ftmac100.c index a31661948c42..9bd7746cbfcf 100644 --- a/drivers/net/ftmac100.c +++ b/drivers/net/ftmac100.c | |||
| @@ -139,11 +139,11 @@ static int ftmac100_reset(struct ftmac100 *priv) | |||
| 139 | * that hardware reset completed (what the f*ck). | 139 | * that hardware reset completed (what the f*ck). |
| 140 | * We still need to wait for a while. | 140 | * We still need to wait for a while. |
| 141 | */ | 141 | */ |
| 142 | usleep_range(500, 1000); | 142 | udelay(500); |
| 143 | return 0; | 143 | return 0; |
| 144 | } | 144 | } |
| 145 | 145 | ||
| 146 | usleep_range(1000, 10000); | 146 | udelay(1000); |
| 147 | } | 147 | } |
| 148 | 148 | ||
| 149 | netdev_err(netdev, "software reset failed\n"); | 149 | netdev_err(netdev, "software reset failed\n"); |
| @@ -772,7 +772,7 @@ static int ftmac100_mdio_read(struct net_device *netdev, int phy_id, int reg) | |||
| 772 | if ((phycr & FTMAC100_PHYCR_MIIRD) == 0) | 772 | if ((phycr & FTMAC100_PHYCR_MIIRD) == 0) |
| 773 | return phycr & FTMAC100_PHYCR_MIIRDATA; | 773 | return phycr & FTMAC100_PHYCR_MIIRDATA; |
| 774 | 774 | ||
| 775 | usleep_range(100, 1000); | 775 | udelay(100); |
| 776 | } | 776 | } |
| 777 | 777 | ||
| 778 | netdev_err(netdev, "mdio read timed out\n"); | 778 | netdev_err(netdev, "mdio read timed out\n"); |
| @@ -801,7 +801,7 @@ static void ftmac100_mdio_write(struct net_device *netdev, int phy_id, int reg, | |||
| 801 | if ((phycr & FTMAC100_PHYCR_MIIWR) == 0) | 801 | if ((phycr & FTMAC100_PHYCR_MIIWR) == 0) |
| 802 | return; | 802 | return; |
| 803 | 803 | ||
| 804 | usleep_range(100, 1000); | 804 | udelay(100); |
| 805 | } | 805 | } |
| 806 | 806 | ||
| 807 | netdev_err(netdev, "mdio write timed out\n"); | 807 | netdev_err(netdev, "mdio write timed out\n"); |
diff --git a/drivers/net/mii.c b/drivers/net/mii.c index 0a6c6a2e7550..d4fc00b1ff93 100644 --- a/drivers/net/mii.c +++ b/drivers/net/mii.c | |||
| @@ -49,6 +49,10 @@ static u32 mii_get_an(struct mii_if_info *mii, u16 addr) | |||
| 49 | result |= ADVERTISED_100baseT_Half; | 49 | result |= ADVERTISED_100baseT_Half; |
| 50 | if (advert & ADVERTISE_100FULL) | 50 | if (advert & ADVERTISE_100FULL) |
| 51 | result |= ADVERTISED_100baseT_Full; | 51 | result |= ADVERTISED_100baseT_Full; |
| 52 | if (advert & ADVERTISE_PAUSE_CAP) | ||
| 53 | result |= ADVERTISED_Pause; | ||
| 54 | if (advert & ADVERTISE_PAUSE_ASYM) | ||
| 55 | result |= ADVERTISED_Asym_Pause; | ||
| 52 | 56 | ||
| 53 | return result; | 57 | return result; |
| 54 | } | 58 | } |
diff --git a/drivers/net/netconsole.c b/drivers/net/netconsole.c index dfb67eb2a94b..eb41e44921e6 100644 --- a/drivers/net/netconsole.c +++ b/drivers/net/netconsole.c | |||
| @@ -671,6 +671,7 @@ static int netconsole_netdev_event(struct notifier_block *this, | |||
| 671 | goto done; | 671 | goto done; |
| 672 | 672 | ||
| 673 | spin_lock_irqsave(&target_list_lock, flags); | 673 | spin_lock_irqsave(&target_list_lock, flags); |
| 674 | restart: | ||
| 674 | list_for_each_entry(nt, &target_list, list) { | 675 | list_for_each_entry(nt, &target_list, list) { |
| 675 | netconsole_target_get(nt); | 676 | netconsole_target_get(nt); |
| 676 | if (nt->np.dev == dev) { | 677 | if (nt->np.dev == dev) { |
| @@ -683,9 +684,16 @@ static int netconsole_netdev_event(struct notifier_block *this, | |||
| 683 | * rtnl_lock already held | 684 | * rtnl_lock already held |
| 684 | */ | 685 | */ |
| 685 | if (nt->np.dev) { | 686 | if (nt->np.dev) { |
| 687 | spin_unlock_irqrestore( | ||
| 688 | &target_list_lock, | ||
| 689 | flags); | ||
| 686 | __netpoll_cleanup(&nt->np); | 690 | __netpoll_cleanup(&nt->np); |
| 691 | spin_lock_irqsave(&target_list_lock, | ||
| 692 | flags); | ||
| 687 | dev_put(nt->np.dev); | 693 | dev_put(nt->np.dev); |
| 688 | nt->np.dev = NULL; | 694 | nt->np.dev = NULL; |
| 695 | netconsole_target_put(nt); | ||
| 696 | goto restart; | ||
| 689 | } | 697 | } |
| 690 | /* Fall through */ | 698 | /* Fall through */ |
| 691 | case NETDEV_GOING_DOWN: | 699 | case NETDEV_GOING_DOWN: |
diff --git a/drivers/net/r8169.c b/drivers/net/r8169.c index 493b0de3848b..397c36810a15 100644 --- a/drivers/net/r8169.c +++ b/drivers/net/r8169.c | |||
| @@ -170,6 +170,16 @@ static const struct { | |||
| 170 | }; | 170 | }; |
| 171 | #undef _R | 171 | #undef _R |
| 172 | 172 | ||
| 173 | static const struct rtl_firmware_info { | ||
| 174 | int mac_version; | ||
| 175 | const char *fw_name; | ||
| 176 | } rtl_firmware_infos[] = { | ||
| 177 | { .mac_version = RTL_GIGA_MAC_VER_25, .fw_name = FIRMWARE_8168D_1 }, | ||
| 178 | { .mac_version = RTL_GIGA_MAC_VER_26, .fw_name = FIRMWARE_8168D_2 }, | ||
| 179 | { .mac_version = RTL_GIGA_MAC_VER_29, .fw_name = FIRMWARE_8105E_1 }, | ||
| 180 | { .mac_version = RTL_GIGA_MAC_VER_30, .fw_name = FIRMWARE_8105E_1 } | ||
| 181 | }; | ||
| 182 | |||
| 173 | enum cfg_version { | 183 | enum cfg_version { |
| 174 | RTL_CFG_0 = 0x00, | 184 | RTL_CFG_0 = 0x00, |
| 175 | RTL_CFG_1, | 185 | RTL_CFG_1, |
| @@ -565,6 +575,7 @@ struct rtl8169_private { | |||
| 565 | u32 saved_wolopts; | 575 | u32 saved_wolopts; |
| 566 | 576 | ||
| 567 | const struct firmware *fw; | 577 | const struct firmware *fw; |
| 578 | #define RTL_FIRMWARE_UNKNOWN ERR_PTR(-EAGAIN); | ||
| 568 | }; | 579 | }; |
| 569 | 580 | ||
| 570 | MODULE_AUTHOR("Realtek and the Linux r8169 crew <netdev@vger.kernel.org>"); | 581 | MODULE_AUTHOR("Realtek and the Linux r8169 crew <netdev@vger.kernel.org>"); |
| @@ -1789,25 +1800,26 @@ rtl_phy_write_fw(struct rtl8169_private *tp, const struct firmware *fw) | |||
| 1789 | 1800 | ||
| 1790 | static void rtl_release_firmware(struct rtl8169_private *tp) | 1801 | static void rtl_release_firmware(struct rtl8169_private *tp) |
| 1791 | { | 1802 | { |
| 1792 | release_firmware(tp->fw); | 1803 | if (!IS_ERR_OR_NULL(tp->fw)) |
| 1793 | tp->fw = NULL; | 1804 | release_firmware(tp->fw); |
| 1805 | tp->fw = RTL_FIRMWARE_UNKNOWN; | ||
| 1794 | } | 1806 | } |
| 1795 | 1807 | ||
| 1796 | static int rtl_apply_firmware(struct rtl8169_private *tp, const char *fw_name) | 1808 | static void rtl_apply_firmware(struct rtl8169_private *tp) |
| 1797 | { | 1809 | { |
| 1798 | const struct firmware **fw = &tp->fw; | 1810 | const struct firmware *fw = tp->fw; |
| 1799 | int rc = !*fw; | ||
| 1800 | |||
| 1801 | if (rc) { | ||
| 1802 | rc = request_firmware(fw, fw_name, &tp->pci_dev->dev); | ||
| 1803 | if (rc < 0) | ||
| 1804 | goto out; | ||
| 1805 | } | ||
| 1806 | 1811 | ||
| 1807 | /* TODO: release firmware once rtl_phy_write_fw signals failures. */ | 1812 | /* TODO: release firmware once rtl_phy_write_fw signals failures. */ |
| 1808 | rtl_phy_write_fw(tp, *fw); | 1813 | if (!IS_ERR_OR_NULL(fw)) |
| 1809 | out: | 1814 | rtl_phy_write_fw(tp, fw); |
| 1810 | return rc; | 1815 | } |
| 1816 | |||
| 1817 | static void rtl_apply_firmware_cond(struct rtl8169_private *tp, u8 reg, u16 val) | ||
| 1818 | { | ||
| 1819 | if (rtl_readphy(tp, reg) != val) | ||
| 1820 | netif_warn(tp, hw, tp->dev, "chipset not ready for firmware\n"); | ||
| 1821 | else | ||
| 1822 | rtl_apply_firmware(tp); | ||
| 1811 | } | 1823 | } |
| 1812 | 1824 | ||
| 1813 | static void rtl8169s_hw_phy_config(struct rtl8169_private *tp) | 1825 | static void rtl8169s_hw_phy_config(struct rtl8169_private *tp) |
| @@ -2246,10 +2258,8 @@ static void rtl8168d_1_hw_phy_config(struct rtl8169_private *tp) | |||
| 2246 | 2258 | ||
| 2247 | rtl_writephy(tp, 0x1f, 0x0005); | 2259 | rtl_writephy(tp, 0x1f, 0x0005); |
| 2248 | rtl_writephy(tp, 0x05, 0x001b); | 2260 | rtl_writephy(tp, 0x05, 0x001b); |
| 2249 | if ((rtl_readphy(tp, 0x06) != 0xbf00) || | 2261 | |
| 2250 | (rtl_apply_firmware(tp, FIRMWARE_8168D_1) < 0)) { | 2262 | rtl_apply_firmware_cond(tp, MII_EXPANSION, 0xbf00); |
| 2251 | netif_warn(tp, probe, tp->dev, "unable to apply firmware patch\n"); | ||
| 2252 | } | ||
| 2253 | 2263 | ||
| 2254 | rtl_writephy(tp, 0x1f, 0x0000); | 2264 | rtl_writephy(tp, 0x1f, 0x0000); |
| 2255 | } | 2265 | } |
| @@ -2351,10 +2361,8 @@ static void rtl8168d_2_hw_phy_config(struct rtl8169_private *tp) | |||
| 2351 | 2361 | ||
| 2352 | rtl_writephy(tp, 0x1f, 0x0005); | 2362 | rtl_writephy(tp, 0x1f, 0x0005); |
| 2353 | rtl_writephy(tp, 0x05, 0x001b); | 2363 | rtl_writephy(tp, 0x05, 0x001b); |
| 2354 | if ((rtl_readphy(tp, 0x06) != 0xb300) || | 2364 | |
| 2355 | (rtl_apply_firmware(tp, FIRMWARE_8168D_2) < 0)) { | 2365 | rtl_apply_firmware_cond(tp, MII_EXPANSION, 0xb300); |
| 2356 | netif_warn(tp, probe, tp->dev, "unable to apply firmware patch\n"); | ||
| 2357 | } | ||
| 2358 | 2366 | ||
| 2359 | rtl_writephy(tp, 0x1f, 0x0000); | 2367 | rtl_writephy(tp, 0x1f, 0x0000); |
| 2360 | } | 2368 | } |
| @@ -2474,8 +2482,7 @@ static void rtl8105e_hw_phy_config(struct rtl8169_private *tp) | |||
| 2474 | rtl_writephy(tp, 0x18, 0x0310); | 2482 | rtl_writephy(tp, 0x18, 0x0310); |
| 2475 | msleep(100); | 2483 | msleep(100); |
| 2476 | 2484 | ||
| 2477 | if (rtl_apply_firmware(tp, FIRMWARE_8105E_1) < 0) | 2485 | rtl_apply_firmware(tp); |
| 2478 | netif_warn(tp, probe, tp->dev, "unable to apply firmware patch\n"); | ||
| 2479 | 2486 | ||
| 2480 | rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init)); | 2487 | rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init)); |
| 2481 | } | 2488 | } |
| @@ -3237,6 +3244,8 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
| 3237 | tp->timer.data = (unsigned long) dev; | 3244 | tp->timer.data = (unsigned long) dev; |
| 3238 | tp->timer.function = rtl8169_phy_timer; | 3245 | tp->timer.function = rtl8169_phy_timer; |
| 3239 | 3246 | ||
| 3247 | tp->fw = RTL_FIRMWARE_UNKNOWN; | ||
| 3248 | |||
| 3240 | rc = register_netdev(dev); | 3249 | rc = register_netdev(dev); |
| 3241 | if (rc < 0) | 3250 | if (rc < 0) |
| 3242 | goto err_out_msi_4; | 3251 | goto err_out_msi_4; |
| @@ -3288,10 +3297,10 @@ static void __devexit rtl8169_remove_one(struct pci_dev *pdev) | |||
| 3288 | 3297 | ||
| 3289 | cancel_delayed_work_sync(&tp->task); | 3298 | cancel_delayed_work_sync(&tp->task); |
| 3290 | 3299 | ||
| 3291 | rtl_release_firmware(tp); | ||
| 3292 | |||
| 3293 | unregister_netdev(dev); | 3300 | unregister_netdev(dev); |
| 3294 | 3301 | ||
| 3302 | rtl_release_firmware(tp); | ||
| 3303 | |||
| 3295 | if (pci_dev_run_wake(pdev)) | 3304 | if (pci_dev_run_wake(pdev)) |
| 3296 | pm_runtime_get_noresume(&pdev->dev); | 3305 | pm_runtime_get_noresume(&pdev->dev); |
| 3297 | 3306 | ||
| @@ -3303,6 +3312,37 @@ static void __devexit rtl8169_remove_one(struct pci_dev *pdev) | |||
| 3303 | pci_set_drvdata(pdev, NULL); | 3312 | pci_set_drvdata(pdev, NULL); |
| 3304 | } | 3313 | } |
| 3305 | 3314 | ||
| 3315 | static void rtl_request_firmware(struct rtl8169_private *tp) | ||
| 3316 | { | ||
| 3317 | int i; | ||
| 3318 | |||
| 3319 | /* Return early if the firmware is already loaded / cached. */ | ||
| 3320 | if (!IS_ERR(tp->fw)) | ||
| 3321 | goto out; | ||
| 3322 | |||
| 3323 | for (i = 0; i < ARRAY_SIZE(rtl_firmware_infos); i++) { | ||
| 3324 | const struct rtl_firmware_info *info = rtl_firmware_infos + i; | ||
| 3325 | |||
| 3326 | if (info->mac_version == tp->mac_version) { | ||
| 3327 | const char *name = info->fw_name; | ||
| 3328 | int rc; | ||
| 3329 | |||
| 3330 | rc = request_firmware(&tp->fw, name, &tp->pci_dev->dev); | ||
| 3331 | if (rc < 0) { | ||
| 3332 | netif_warn(tp, ifup, tp->dev, "unable to load " | ||
| 3333 | "firmware patch %s (%d)\n", name, rc); | ||
| 3334 | goto out_disable_request_firmware; | ||
| 3335 | } | ||
| 3336 | goto out; | ||
| 3337 | } | ||
| 3338 | } | ||
| 3339 | |||
| 3340 | out_disable_request_firmware: | ||
| 3341 | tp->fw = NULL; | ||
| 3342 | out: | ||
| 3343 | return; | ||
| 3344 | } | ||
| 3345 | |||
| 3306 | static int rtl8169_open(struct net_device *dev) | 3346 | static int rtl8169_open(struct net_device *dev) |
| 3307 | { | 3347 | { |
| 3308 | struct rtl8169_private *tp = netdev_priv(dev); | 3348 | struct rtl8169_private *tp = netdev_priv(dev); |
| @@ -3334,11 +3374,13 @@ static int rtl8169_open(struct net_device *dev) | |||
| 3334 | 3374 | ||
| 3335 | smp_mb(); | 3375 | smp_mb(); |
| 3336 | 3376 | ||
| 3377 | rtl_request_firmware(tp); | ||
| 3378 | |||
| 3337 | retval = request_irq(dev->irq, rtl8169_interrupt, | 3379 | retval = request_irq(dev->irq, rtl8169_interrupt, |
| 3338 | (tp->features & RTL_FEATURE_MSI) ? 0 : IRQF_SHARED, | 3380 | (tp->features & RTL_FEATURE_MSI) ? 0 : IRQF_SHARED, |
| 3339 | dev->name, dev); | 3381 | dev->name, dev); |
| 3340 | if (retval < 0) | 3382 | if (retval < 0) |
| 3341 | goto err_release_ring_2; | 3383 | goto err_release_fw_2; |
| 3342 | 3384 | ||
| 3343 | napi_enable(&tp->napi); | 3385 | napi_enable(&tp->napi); |
| 3344 | 3386 | ||
| @@ -3359,7 +3401,8 @@ static int rtl8169_open(struct net_device *dev) | |||
| 3359 | out: | 3401 | out: |
| 3360 | return retval; | 3402 | return retval; |
| 3361 | 3403 | ||
| 3362 | err_release_ring_2: | 3404 | err_release_fw_2: |
| 3405 | rtl_release_firmware(tp); | ||
| 3363 | rtl8169_rx_clear(tp); | 3406 | rtl8169_rx_clear(tp); |
| 3364 | err_free_rx_1: | 3407 | err_free_rx_1: |
| 3365 | dma_free_coherent(&pdev->dev, R8169_RX_RING_BYTES, tp->RxDescArray, | 3408 | dma_free_coherent(&pdev->dev, R8169_RX_RING_BYTES, tp->RxDescArray, |
diff --git a/drivers/net/tg3.c b/drivers/net/tg3.c index b8c5f35577e4..7a5daefb6f33 100644 --- a/drivers/net/tg3.c +++ b/drivers/net/tg3.c | |||
| @@ -12327,8 +12327,10 @@ static void __devinit tg3_get_eeprom_hw_cfg(struct tg3 *tp) | |||
| 12327 | if (val & VCPU_CFGSHDW_ASPM_DBNC) | 12327 | if (val & VCPU_CFGSHDW_ASPM_DBNC) |
| 12328 | tp->tg3_flags |= TG3_FLAG_ASPM_WORKAROUND; | 12328 | tp->tg3_flags |= TG3_FLAG_ASPM_WORKAROUND; |
| 12329 | if ((val & VCPU_CFGSHDW_WOL_ENABLE) && | 12329 | if ((val & VCPU_CFGSHDW_WOL_ENABLE) && |
| 12330 | (val & VCPU_CFGSHDW_WOL_MAGPKT)) | 12330 | (val & VCPU_CFGSHDW_WOL_MAGPKT)) { |
| 12331 | tp->tg3_flags |= TG3_FLAG_WOL_ENABLE; | 12331 | tp->tg3_flags |= TG3_FLAG_WOL_ENABLE; |
| 12332 | device_set_wakeup_enable(&tp->pdev->dev, true); | ||
| 12333 | } | ||
| 12332 | goto done; | 12334 | goto done; |
| 12333 | } | 12335 | } |
| 12334 | 12336 | ||
| @@ -12461,8 +12463,10 @@ static void __devinit tg3_get_eeprom_hw_cfg(struct tg3 *tp) | |||
| 12461 | tp->tg3_flags &= ~TG3_FLAG_WOL_CAP; | 12463 | tp->tg3_flags &= ~TG3_FLAG_WOL_CAP; |
| 12462 | 12464 | ||
| 12463 | if ((tp->tg3_flags & TG3_FLAG_WOL_CAP) && | 12465 | if ((tp->tg3_flags & TG3_FLAG_WOL_CAP) && |
| 12464 | (nic_cfg & NIC_SRAM_DATA_CFG_WOL_ENABLE)) | 12466 | (nic_cfg & NIC_SRAM_DATA_CFG_WOL_ENABLE)) { |
| 12465 | tp->tg3_flags |= TG3_FLAG_WOL_ENABLE; | 12467 | tp->tg3_flags |= TG3_FLAG_WOL_ENABLE; |
| 12468 | device_set_wakeup_enable(&tp->pdev->dev, true); | ||
| 12469 | } | ||
| 12466 | 12470 | ||
| 12467 | if (cfg2 & (1 << 17)) | 12471 | if (cfg2 & (1 << 17)) |
| 12468 | tp->phy_flags |= TG3_PHYFLG_CAPACITIVE_COUPLING; | 12472 | tp->phy_flags |= TG3_PHYFLG_CAPACITIVE_COUPLING; |
diff --git a/drivers/net/usb/cdc_ether.c b/drivers/net/usb/cdc_ether.c index 341f7056a800..a301479ecc60 100644 --- a/drivers/net/usb/cdc_ether.c +++ b/drivers/net/usb/cdc_ether.c | |||
| @@ -460,7 +460,7 @@ static const struct driver_info cdc_info = { | |||
| 460 | .manage_power = cdc_manage_power, | 460 | .manage_power = cdc_manage_power, |
| 461 | }; | 461 | }; |
| 462 | 462 | ||
| 463 | static const struct driver_info mbm_info = { | 463 | static const struct driver_info wwan_info = { |
| 464 | .description = "Mobile Broadband Network Device", | 464 | .description = "Mobile Broadband Network Device", |
| 465 | .flags = FLAG_WWAN, | 465 | .flags = FLAG_WWAN, |
| 466 | .bind = usbnet_cdc_bind, | 466 | .bind = usbnet_cdc_bind, |
| @@ -471,6 +471,7 @@ static const struct driver_info mbm_info = { | |||
| 471 | 471 | ||
| 472 | /*-------------------------------------------------------------------------*/ | 472 | /*-------------------------------------------------------------------------*/ |
| 473 | 473 | ||
| 474 | #define HUAWEI_VENDOR_ID 0x12D1 | ||
| 474 | 475 | ||
| 475 | static const struct usb_device_id products [] = { | 476 | static const struct usb_device_id products [] = { |
| 476 | /* | 477 | /* |
| @@ -587,8 +588,17 @@ static const struct usb_device_id products [] = { | |||
| 587 | }, { | 588 | }, { |
| 588 | USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_MDLM, | 589 | USB_INTERFACE_INFO(USB_CLASS_COMM, USB_CDC_SUBCLASS_MDLM, |
| 589 | USB_CDC_PROTO_NONE), | 590 | USB_CDC_PROTO_NONE), |
| 590 | .driver_info = (unsigned long)&mbm_info, | 591 | .driver_info = (unsigned long)&wwan_info, |
| 591 | 592 | ||
| 593 | }, { | ||
| 594 | /* Various Huawei modems with a network port like the UMG1831 */ | ||
| 595 | .match_flags = USB_DEVICE_ID_MATCH_VENDOR | ||
| 596 | | USB_DEVICE_ID_MATCH_INT_INFO, | ||
| 597 | .idVendor = HUAWEI_VENDOR_ID, | ||
| 598 | .bInterfaceClass = USB_CLASS_COMM, | ||
| 599 | .bInterfaceSubClass = USB_CDC_SUBCLASS_ETHERNET, | ||
| 600 | .bInterfaceProtocol = 255, | ||
| 601 | .driver_info = (unsigned long)&wwan_info, | ||
| 592 | }, | 602 | }, |
| 593 | { }, // END | 603 | { }, // END |
| 594 | }; | 604 | }; |
diff --git a/drivers/net/usb/cdc_ncm.c b/drivers/net/usb/cdc_ncm.c index 967371f04454..1033ef6476a4 100644 --- a/drivers/net/usb/cdc_ncm.c +++ b/drivers/net/usb/cdc_ncm.c | |||
| @@ -54,13 +54,13 @@ | |||
| 54 | #include <linux/usb/usbnet.h> | 54 | #include <linux/usb/usbnet.h> |
| 55 | #include <linux/usb/cdc.h> | 55 | #include <linux/usb/cdc.h> |
| 56 | 56 | ||
| 57 | #define DRIVER_VERSION "7-Feb-2011" | 57 | #define DRIVER_VERSION "23-Apr-2011" |
| 58 | 58 | ||
| 59 | /* CDC NCM subclass 3.2.1 */ | 59 | /* CDC NCM subclass 3.2.1 */ |
| 60 | #define USB_CDC_NCM_NDP16_LENGTH_MIN 0x10 | 60 | #define USB_CDC_NCM_NDP16_LENGTH_MIN 0x10 |
| 61 | 61 | ||
| 62 | /* Maximum NTB length */ | 62 | /* Maximum NTB length */ |
| 63 | #define CDC_NCM_NTB_MAX_SIZE_TX 16384 /* bytes */ | 63 | #define CDC_NCM_NTB_MAX_SIZE_TX (16384 + 4) /* bytes, must be short terminated */ |
| 64 | #define CDC_NCM_NTB_MAX_SIZE_RX 16384 /* bytes */ | 64 | #define CDC_NCM_NTB_MAX_SIZE_RX 16384 /* bytes */ |
| 65 | 65 | ||
| 66 | /* Minimum value for MaxDatagramSize, ch. 6.2.9 */ | 66 | /* Minimum value for MaxDatagramSize, ch. 6.2.9 */ |
diff --git a/drivers/net/usb/smsc95xx.c b/drivers/net/usb/smsc95xx.c index 47a6c870b51f..48d4efdb4959 100644 --- a/drivers/net/usb/smsc95xx.c +++ b/drivers/net/usb/smsc95xx.c | |||
| @@ -730,7 +730,7 @@ static int smsc95xx_phy_initialize(struct usbnet *dev) | |||
| 730 | msleep(10); | 730 | msleep(10); |
| 731 | bmcr = smsc95xx_mdio_read(dev->net, dev->mii.phy_id, MII_BMCR); | 731 | bmcr = smsc95xx_mdio_read(dev->net, dev->mii.phy_id, MII_BMCR); |
| 732 | timeout++; | 732 | timeout++; |
| 733 | } while ((bmcr & MII_BMCR) && (timeout < 100)); | 733 | } while ((bmcr & BMCR_RESET) && (timeout < 100)); |
| 734 | 734 | ||
| 735 | if (timeout >= 100) { | 735 | if (timeout >= 100) { |
| 736 | netdev_warn(dev->net, "timeout on PHY Reset"); | 736 | netdev_warn(dev->net, "timeout on PHY Reset"); |
diff --git a/drivers/net/usb/usbnet.c b/drivers/net/usb/usbnet.c index 069c1cf0fdf7..009bba3d753e 100644 --- a/drivers/net/usb/usbnet.c +++ b/drivers/net/usb/usbnet.c | |||
| @@ -736,6 +736,7 @@ int usbnet_open (struct net_device *net) | |||
| 736 | } | 736 | } |
| 737 | } | 737 | } |
| 738 | 738 | ||
| 739 | set_bit(EVENT_DEV_OPEN, &dev->flags); | ||
| 739 | netif_start_queue (net); | 740 | netif_start_queue (net); |
| 740 | netif_info(dev, ifup, dev->net, | 741 | netif_info(dev, ifup, dev->net, |
| 741 | "open: enable queueing (rx %d, tx %d) mtu %d %s framing\n", | 742 | "open: enable queueing (rx %d, tx %d) mtu %d %s framing\n", |
| @@ -1259,6 +1260,9 @@ void usbnet_disconnect (struct usb_interface *intf) | |||
| 1259 | if (dev->driver_info->unbind) | 1260 | if (dev->driver_info->unbind) |
| 1260 | dev->driver_info->unbind (dev, intf); | 1261 | dev->driver_info->unbind (dev, intf); |
| 1261 | 1262 | ||
| 1263 | usb_kill_urb(dev->interrupt); | ||
| 1264 | usb_free_urb(dev->interrupt); | ||
| 1265 | |||
| 1262 | free_netdev(net); | 1266 | free_netdev(net); |
| 1263 | usb_put_dev (xdev); | 1267 | usb_put_dev (xdev); |
| 1264 | } | 1268 | } |
| @@ -1498,6 +1502,10 @@ int usbnet_resume (struct usb_interface *intf) | |||
| 1498 | int retval; | 1502 | int retval; |
| 1499 | 1503 | ||
| 1500 | if (!--dev->suspend_count) { | 1504 | if (!--dev->suspend_count) { |
| 1505 | /* resume interrupt URBs */ | ||
| 1506 | if (dev->interrupt && test_bit(EVENT_DEV_OPEN, &dev->flags)) | ||
| 1507 | usb_submit_urb(dev->interrupt, GFP_NOIO); | ||
| 1508 | |||
| 1501 | spin_lock_irq(&dev->txq.lock); | 1509 | spin_lock_irq(&dev->txq.lock); |
| 1502 | while ((res = usb_get_from_anchor(&dev->deferred))) { | 1510 | while ((res = usb_get_from_anchor(&dev->deferred))) { |
| 1503 | 1511 | ||
diff --git a/drivers/net/veth.c b/drivers/net/veth.c index 2de9b90c5f8f..3b99f64104fd 100644 --- a/drivers/net/veth.c +++ b/drivers/net/veth.c | |||
| @@ -403,6 +403,17 @@ static int veth_newlink(struct net *src_net, struct net_device *dev, | |||
| 403 | if (tb[IFLA_ADDRESS] == NULL) | 403 | if (tb[IFLA_ADDRESS] == NULL) |
| 404 | random_ether_addr(dev->dev_addr); | 404 | random_ether_addr(dev->dev_addr); |
| 405 | 405 | ||
| 406 | if (tb[IFLA_IFNAME]) | ||
| 407 | nla_strlcpy(dev->name, tb[IFLA_IFNAME], IFNAMSIZ); | ||
| 408 | else | ||
| 409 | snprintf(dev->name, IFNAMSIZ, DRV_NAME "%%d"); | ||
| 410 | |||
| 411 | if (strchr(dev->name, '%')) { | ||
| 412 | err = dev_alloc_name(dev, dev->name); | ||
| 413 | if (err < 0) | ||
| 414 | goto err_alloc_name; | ||
| 415 | } | ||
| 416 | |||
| 406 | err = register_netdevice(dev); | 417 | err = register_netdevice(dev); |
| 407 | if (err < 0) | 418 | if (err < 0) |
| 408 | goto err_register_dev; | 419 | goto err_register_dev; |
| @@ -422,6 +433,7 @@ static int veth_newlink(struct net *src_net, struct net_device *dev, | |||
| 422 | 433 | ||
| 423 | err_register_dev: | 434 | err_register_dev: |
| 424 | /* nothing to do */ | 435 | /* nothing to do */ |
| 436 | err_alloc_name: | ||
| 425 | err_configure_peer: | 437 | err_configure_peer: |
| 426 | unregister_netdevice(peer); | 438 | unregister_netdevice(peer); |
| 427 | return err; | 439 | return err; |
diff --git a/drivers/net/wireless/ath/ath9k/recv.c b/drivers/net/wireless/ath/ath9k/recv.c index dcd19bc337d1..b29c80def35e 100644 --- a/drivers/net/wireless/ath/ath9k/recv.c +++ b/drivers/net/wireless/ath/ath9k/recv.c | |||
| @@ -506,7 +506,7 @@ bool ath_stoprecv(struct ath_softc *sc) | |||
| 506 | "confusing the DMA engine when we start RX up\n"); | 506 | "confusing the DMA engine when we start RX up\n"); |
| 507 | ATH_DBG_WARN_ON_ONCE(!stopped); | 507 | ATH_DBG_WARN_ON_ONCE(!stopped); |
| 508 | } | 508 | } |
| 509 | return stopped || reset; | 509 | return stopped && !reset; |
| 510 | } | 510 | } |
| 511 | 511 | ||
| 512 | void ath_flushrecv(struct ath_softc *sc) | 512 | void ath_flushrecv(struct ath_softc *sc) |
diff --git a/drivers/net/wireless/b43/main.c b/drivers/net/wireless/b43/main.c index d59b0168c14a..5af40d9170a0 100644 --- a/drivers/net/wireless/b43/main.c +++ b/drivers/net/wireless/b43/main.c | |||
| @@ -72,6 +72,7 @@ MODULE_FIRMWARE("b43/ucode11.fw"); | |||
| 72 | MODULE_FIRMWARE("b43/ucode13.fw"); | 72 | MODULE_FIRMWARE("b43/ucode13.fw"); |
| 73 | MODULE_FIRMWARE("b43/ucode14.fw"); | 73 | MODULE_FIRMWARE("b43/ucode14.fw"); |
| 74 | MODULE_FIRMWARE("b43/ucode15.fw"); | 74 | MODULE_FIRMWARE("b43/ucode15.fw"); |
| 75 | MODULE_FIRMWARE("b43/ucode16_mimo.fw"); | ||
| 75 | MODULE_FIRMWARE("b43/ucode5.fw"); | 76 | MODULE_FIRMWARE("b43/ucode5.fw"); |
| 76 | MODULE_FIRMWARE("b43/ucode9.fw"); | 77 | MODULE_FIRMWARE("b43/ucode9.fw"); |
| 77 | 78 | ||
diff --git a/drivers/net/wireless/iwlegacy/iwl-4965-tx.c b/drivers/net/wireless/iwlegacy/iwl-4965-tx.c index 5c40502f869a..79ac081832fb 100644 --- a/drivers/net/wireless/iwlegacy/iwl-4965-tx.c +++ b/drivers/net/wireless/iwlegacy/iwl-4965-tx.c | |||
| @@ -316,12 +316,18 @@ int iwl4965_tx_skb(struct iwl_priv *priv, struct sk_buff *skb) | |||
| 316 | 316 | ||
| 317 | hdr_len = ieee80211_hdrlen(fc); | 317 | hdr_len = ieee80211_hdrlen(fc); |
| 318 | 318 | ||
| 319 | /* Find index into station table for destination station */ | 319 | /* For management frames use broadcast id to do not break aggregation */ |
| 320 | sta_id = iwl_legacy_sta_id_or_broadcast(priv, ctx, info->control.sta); | 320 | if (!ieee80211_is_data(fc)) |
| 321 | if (sta_id == IWL_INVALID_STATION) { | 321 | sta_id = ctx->bcast_sta_id; |
| 322 | IWL_DEBUG_DROP(priv, "Dropping - INVALID STATION: %pM\n", | 322 | else { |
| 323 | hdr->addr1); | 323 | /* Find index into station table for destination station */ |
| 324 | goto drop_unlock; | 324 | sta_id = iwl_legacy_sta_id_or_broadcast(priv, ctx, info->control.sta); |
| 325 | |||
| 326 | if (sta_id == IWL_INVALID_STATION) { | ||
| 327 | IWL_DEBUG_DROP(priv, "Dropping - INVALID STATION: %pM\n", | ||
| 328 | hdr->addr1); | ||
| 329 | goto drop_unlock; | ||
| 330 | } | ||
| 325 | } | 331 | } |
| 326 | 332 | ||
| 327 | IWL_DEBUG_TX(priv, "station Id %d\n", sta_id); | 333 | IWL_DEBUG_TX(priv, "station Id %d\n", sta_id); |
| @@ -1127,12 +1133,16 @@ int iwl4965_tx_queue_reclaim(struct iwl_priv *priv, int txq_id, int index) | |||
| 1127 | q->read_ptr = iwl_legacy_queue_inc_wrap(q->read_ptr, q->n_bd)) { | 1133 | q->read_ptr = iwl_legacy_queue_inc_wrap(q->read_ptr, q->n_bd)) { |
| 1128 | 1134 | ||
| 1129 | tx_info = &txq->txb[txq->q.read_ptr]; | 1135 | tx_info = &txq->txb[txq->q.read_ptr]; |
| 1130 | iwl4965_tx_status(priv, tx_info, | 1136 | |
| 1131 | txq_id >= IWL4965_FIRST_AMPDU_QUEUE); | 1137 | if (WARN_ON_ONCE(tx_info->skb == NULL)) |
| 1138 | continue; | ||
| 1132 | 1139 | ||
| 1133 | hdr = (struct ieee80211_hdr *)tx_info->skb->data; | 1140 | hdr = (struct ieee80211_hdr *)tx_info->skb->data; |
| 1134 | if (hdr && ieee80211_is_data_qos(hdr->frame_control)) | 1141 | if (ieee80211_is_data_qos(hdr->frame_control)) |
| 1135 | nfreed++; | 1142 | nfreed++; |
| 1143 | |||
| 1144 | iwl4965_tx_status(priv, tx_info, | ||
| 1145 | txq_id >= IWL4965_FIRST_AMPDU_QUEUE); | ||
| 1136 | tx_info->skb = NULL; | 1146 | tx_info->skb = NULL; |
| 1137 | 1147 | ||
| 1138 | priv->cfg->ops->lib->txq_free_tfd(priv, txq); | 1148 | priv->cfg->ops->lib->txq_free_tfd(priv, txq); |
diff --git a/drivers/net/wireless/iwlegacy/iwl-led.c b/drivers/net/wireless/iwlegacy/iwl-led.c index 15eb8b707157..bda0d61b2c0d 100644 --- a/drivers/net/wireless/iwlegacy/iwl-led.c +++ b/drivers/net/wireless/iwlegacy/iwl-led.c | |||
| @@ -48,8 +48,21 @@ module_param(led_mode, int, S_IRUGO); | |||
| 48 | MODULE_PARM_DESC(led_mode, "0=system default, " | 48 | MODULE_PARM_DESC(led_mode, "0=system default, " |
| 49 | "1=On(RF On)/Off(RF Off), 2=blinking"); | 49 | "1=On(RF On)/Off(RF Off), 2=blinking"); |
| 50 | 50 | ||
| 51 | /* Throughput OFF time(ms) ON time (ms) | ||
| 52 | * >300 25 25 | ||
| 53 | * >200 to 300 40 40 | ||
| 54 | * >100 to 200 55 55 | ||
| 55 | * >70 to 100 65 65 | ||
| 56 | * >50 to 70 75 75 | ||
| 57 | * >20 to 50 85 85 | ||
| 58 | * >10 to 20 95 95 | ||
| 59 | * >5 to 10 110 110 | ||
| 60 | * >1 to 5 130 130 | ||
| 61 | * >0 to 1 167 167 | ||
| 62 | * <=0 SOLID ON | ||
| 63 | */ | ||
| 51 | static const struct ieee80211_tpt_blink iwl_blink[] = { | 64 | static const struct ieee80211_tpt_blink iwl_blink[] = { |
| 52 | { .throughput = 0 * 1024 - 1, .blink_time = 334 }, | 65 | { .throughput = 0, .blink_time = 334 }, |
| 53 | { .throughput = 1 * 1024 - 1, .blink_time = 260 }, | 66 | { .throughput = 1 * 1024 - 1, .blink_time = 260 }, |
| 54 | { .throughput = 5 * 1024 - 1, .blink_time = 220 }, | 67 | { .throughput = 5 * 1024 - 1, .blink_time = 220 }, |
| 55 | { .throughput = 10 * 1024 - 1, .blink_time = 190 }, | 68 | { .throughput = 10 * 1024 - 1, .blink_time = 190 }, |
| @@ -101,6 +114,11 @@ static int iwl_legacy_led_cmd(struct iwl_priv *priv, | |||
| 101 | if (priv->blink_on == on && priv->blink_off == off) | 114 | if (priv->blink_on == on && priv->blink_off == off) |
| 102 | return 0; | 115 | return 0; |
| 103 | 116 | ||
| 117 | if (off == 0) { | ||
| 118 | /* led is SOLID_ON */ | ||
| 119 | on = IWL_LED_SOLID; | ||
| 120 | } | ||
| 121 | |||
| 104 | IWL_DEBUG_LED(priv, "Led blink time compensation=%u\n", | 122 | IWL_DEBUG_LED(priv, "Led blink time compensation=%u\n", |
| 105 | priv->cfg->base_params->led_compensation); | 123 | priv->cfg->base_params->led_compensation); |
| 106 | led_cmd.on = iwl_legacy_blink_compensation(priv, on, | 124 | led_cmd.on = iwl_legacy_blink_compensation(priv, on, |
diff --git a/drivers/net/wireless/iwlegacy/iwl4965-base.c b/drivers/net/wireless/iwlegacy/iwl4965-base.c index d484c3678163..a62fe24ee594 100644 --- a/drivers/net/wireless/iwlegacy/iwl4965-base.c +++ b/drivers/net/wireless/iwlegacy/iwl4965-base.c | |||
| @@ -2984,15 +2984,15 @@ static void iwl4965_bg_txpower_work(struct work_struct *work) | |||
| 2984 | struct iwl_priv *priv = container_of(work, struct iwl_priv, | 2984 | struct iwl_priv *priv = container_of(work, struct iwl_priv, |
| 2985 | txpower_work); | 2985 | txpower_work); |
| 2986 | 2986 | ||
| 2987 | mutex_lock(&priv->mutex); | ||
| 2988 | |||
| 2987 | /* If a scan happened to start before we got here | 2989 | /* If a scan happened to start before we got here |
| 2988 | * then just return; the statistics notification will | 2990 | * then just return; the statistics notification will |
| 2989 | * kick off another scheduled work to compensate for | 2991 | * kick off another scheduled work to compensate for |
| 2990 | * any temperature delta we missed here. */ | 2992 | * any temperature delta we missed here. */ |
| 2991 | if (test_bit(STATUS_EXIT_PENDING, &priv->status) || | 2993 | if (test_bit(STATUS_EXIT_PENDING, &priv->status) || |
| 2992 | test_bit(STATUS_SCANNING, &priv->status)) | 2994 | test_bit(STATUS_SCANNING, &priv->status)) |
| 2993 | return; | 2995 | goto out; |
| 2994 | |||
| 2995 | mutex_lock(&priv->mutex); | ||
| 2996 | 2996 | ||
| 2997 | /* Regardless of if we are associated, we must reconfigure the | 2997 | /* Regardless of if we are associated, we must reconfigure the |
| 2998 | * TX power since frames can be sent on non-radar channels while | 2998 | * TX power since frames can be sent on non-radar channels while |
| @@ -3002,7 +3002,7 @@ static void iwl4965_bg_txpower_work(struct work_struct *work) | |||
| 3002 | /* Update last_temperature to keep is_calib_needed from running | 3002 | /* Update last_temperature to keep is_calib_needed from running |
| 3003 | * when it isn't needed... */ | 3003 | * when it isn't needed... */ |
| 3004 | priv->last_temperature = priv->temperature; | 3004 | priv->last_temperature = priv->temperature; |
| 3005 | 3005 | out: | |
| 3006 | mutex_unlock(&priv->mutex); | 3006 | mutex_unlock(&priv->mutex); |
| 3007 | } | 3007 | } |
| 3008 | 3008 | ||
diff --git a/drivers/net/wireless/iwlwifi/iwl-agn-rxon.c b/drivers/net/wireless/iwlwifi/iwl-agn-rxon.c index dfdbea6e8f99..fbbde0712fa5 100644 --- a/drivers/net/wireless/iwlwifi/iwl-agn-rxon.c +++ b/drivers/net/wireless/iwlwifi/iwl-agn-rxon.c | |||
| @@ -335,7 +335,6 @@ int iwlagn_mac_config(struct ieee80211_hw *hw, u32 changed) | |||
| 335 | struct ieee80211_channel *channel = conf->channel; | 335 | struct ieee80211_channel *channel = conf->channel; |
| 336 | const struct iwl_channel_info *ch_info; | 336 | const struct iwl_channel_info *ch_info; |
| 337 | int ret = 0; | 337 | int ret = 0; |
| 338 | bool ht_changed[NUM_IWL_RXON_CTX] = {}; | ||
| 339 | 338 | ||
| 340 | IWL_DEBUG_MAC80211(priv, "changed %#x", changed); | 339 | IWL_DEBUG_MAC80211(priv, "changed %#x", changed); |
| 341 | 340 | ||
| @@ -383,10 +382,8 @@ int iwlagn_mac_config(struct ieee80211_hw *hw, u32 changed) | |||
| 383 | 382 | ||
| 384 | for_each_context(priv, ctx) { | 383 | for_each_context(priv, ctx) { |
| 385 | /* Configure HT40 channels */ | 384 | /* Configure HT40 channels */ |
| 386 | if (ctx->ht.enabled != conf_is_ht(conf)) { | 385 | if (ctx->ht.enabled != conf_is_ht(conf)) |
| 387 | ctx->ht.enabled = conf_is_ht(conf); | 386 | ctx->ht.enabled = conf_is_ht(conf); |
| 388 | ht_changed[ctx->ctxid] = true; | ||
| 389 | } | ||
| 390 | 387 | ||
| 391 | if (ctx->ht.enabled) { | 388 | if (ctx->ht.enabled) { |
| 392 | if (conf_is_ht40_minus(conf)) { | 389 | if (conf_is_ht40_minus(conf)) { |
| @@ -455,8 +452,6 @@ int iwlagn_mac_config(struct ieee80211_hw *hw, u32 changed) | |||
| 455 | if (!memcmp(&ctx->staging, &ctx->active, sizeof(ctx->staging))) | 452 | if (!memcmp(&ctx->staging, &ctx->active, sizeof(ctx->staging))) |
| 456 | continue; | 453 | continue; |
| 457 | iwlagn_commit_rxon(priv, ctx); | 454 | iwlagn_commit_rxon(priv, ctx); |
| 458 | if (ht_changed[ctx->ctxid]) | ||
| 459 | iwlagn_update_qos(priv, ctx); | ||
| 460 | } | 455 | } |
| 461 | out: | 456 | out: |
| 462 | mutex_unlock(&priv->mutex); | 457 | mutex_unlock(&priv->mutex); |
diff --git a/drivers/net/wireless/iwlwifi/iwl-agn-tx.c b/drivers/net/wireless/iwlwifi/iwl-agn-tx.c index a709d05c5868..0712b67283a4 100644 --- a/drivers/net/wireless/iwlwifi/iwl-agn-tx.c +++ b/drivers/net/wireless/iwlwifi/iwl-agn-tx.c | |||
| @@ -568,12 +568,17 @@ int iwlagn_tx_skb(struct iwl_priv *priv, struct sk_buff *skb) | |||
| 568 | 568 | ||
| 569 | hdr_len = ieee80211_hdrlen(fc); | 569 | hdr_len = ieee80211_hdrlen(fc); |
| 570 | 570 | ||
| 571 | /* Find index into station table for destination station */ | 571 | /* For management frames use broadcast id to do not break aggregation */ |
| 572 | sta_id = iwl_sta_id_or_broadcast(priv, ctx, info->control.sta); | 572 | if (!ieee80211_is_data(fc)) |
| 573 | if (sta_id == IWL_INVALID_STATION) { | 573 | sta_id = ctx->bcast_sta_id; |
| 574 | IWL_DEBUG_DROP(priv, "Dropping - INVALID STATION: %pM\n", | 574 | else { |
| 575 | hdr->addr1); | 575 | /* Find index into station table for destination station */ |
| 576 | goto drop_unlock; | 576 | sta_id = iwl_sta_id_or_broadcast(priv, ctx, info->control.sta); |
| 577 | if (sta_id == IWL_INVALID_STATION) { | ||
| 578 | IWL_DEBUG_DROP(priv, "Dropping - INVALID STATION: %pM\n", | ||
| 579 | hdr->addr1); | ||
| 580 | goto drop_unlock; | ||
| 581 | } | ||
| 577 | } | 582 | } |
| 578 | 583 | ||
| 579 | IWL_DEBUG_TX(priv, "station Id %d\n", sta_id); | 584 | IWL_DEBUG_TX(priv, "station Id %d\n", sta_id); |
| @@ -1224,12 +1229,16 @@ int iwlagn_tx_queue_reclaim(struct iwl_priv *priv, int txq_id, int index) | |||
| 1224 | q->read_ptr = iwl_queue_inc_wrap(q->read_ptr, q->n_bd)) { | 1229 | q->read_ptr = iwl_queue_inc_wrap(q->read_ptr, q->n_bd)) { |
| 1225 | 1230 | ||
| 1226 | tx_info = &txq->txb[txq->q.read_ptr]; | 1231 | tx_info = &txq->txb[txq->q.read_ptr]; |
| 1227 | iwlagn_tx_status(priv, tx_info, | 1232 | |
| 1228 | txq_id >= IWLAGN_FIRST_AMPDU_QUEUE); | 1233 | if (WARN_ON_ONCE(tx_info->skb == NULL)) |
| 1234 | continue; | ||
| 1229 | 1235 | ||
| 1230 | hdr = (struct ieee80211_hdr *)tx_info->skb->data; | 1236 | hdr = (struct ieee80211_hdr *)tx_info->skb->data; |
| 1231 | if (hdr && ieee80211_is_data_qos(hdr->frame_control)) | 1237 | if (ieee80211_is_data_qos(hdr->frame_control)) |
| 1232 | nfreed++; | 1238 | nfreed++; |
| 1239 | |||
| 1240 | iwlagn_tx_status(priv, tx_info, | ||
| 1241 | txq_id >= IWLAGN_FIRST_AMPDU_QUEUE); | ||
| 1233 | tx_info->skb = NULL; | 1242 | tx_info->skb = NULL; |
| 1234 | 1243 | ||
| 1235 | if (priv->cfg->ops->lib->txq_inval_byte_cnt_tbl) | 1244 | if (priv->cfg->ops->lib->txq_inval_byte_cnt_tbl) |
diff --git a/drivers/pcmcia/pcmcia_resource.c b/drivers/pcmcia/pcmcia_resource.c index fe77e8223841..e8c19def1b0f 100644 --- a/drivers/pcmcia/pcmcia_resource.c +++ b/drivers/pcmcia/pcmcia_resource.c | |||
| @@ -173,7 +173,7 @@ static int pcmcia_access_config(struct pcmcia_device *p_dev, | |||
| 173 | c = p_dev->function_config; | 173 | c = p_dev->function_config; |
| 174 | 174 | ||
| 175 | if (!(c->state & CONFIG_LOCKED)) { | 175 | if (!(c->state & CONFIG_LOCKED)) { |
| 176 | dev_dbg(&p_dev->dev, "Configuration isn't't locked\n"); | 176 | dev_dbg(&p_dev->dev, "Configuration isn't locked\n"); |
| 177 | mutex_unlock(&s->ops_mutex); | 177 | mutex_unlock(&s->ops_mutex); |
| 178 | return -EACCES; | 178 | return -EACCES; |
| 179 | } | 179 | } |
diff --git a/drivers/rtc/rtc-max8925.c b/drivers/rtc/rtc-max8925.c index 174036dda786..20494b5edc3c 100644 --- a/drivers/rtc/rtc-max8925.c +++ b/drivers/rtc/rtc-max8925.c | |||
| @@ -257,6 +257,8 @@ static int __devinit max8925_rtc_probe(struct platform_device *pdev) | |||
| 257 | goto out_irq; | 257 | goto out_irq; |
| 258 | } | 258 | } |
| 259 | 259 | ||
| 260 | dev_set_drvdata(&pdev->dev, info); | ||
| 261 | |||
| 260 | info->rtc_dev = rtc_device_register("max8925-rtc", &pdev->dev, | 262 | info->rtc_dev = rtc_device_register("max8925-rtc", &pdev->dev, |
| 261 | &max8925_rtc_ops, THIS_MODULE); | 263 | &max8925_rtc_ops, THIS_MODULE); |
| 262 | ret = PTR_ERR(info->rtc_dev); | 264 | ret = PTR_ERR(info->rtc_dev); |
| @@ -265,7 +267,6 @@ static int __devinit max8925_rtc_probe(struct platform_device *pdev) | |||
| 265 | goto out_rtc; | 267 | goto out_rtc; |
| 266 | } | 268 | } |
| 267 | 269 | ||
| 268 | dev_set_drvdata(&pdev->dev, info); | ||
| 269 | platform_set_drvdata(pdev, info); | 270 | platform_set_drvdata(pdev, info); |
| 270 | 271 | ||
| 271 | return 0; | 272 | return 0; |
diff --git a/drivers/s390/block/dasd_diag.c b/drivers/s390/block/dasd_diag.c index 29143eda9dd9..85dddb1e4126 100644 --- a/drivers/s390/block/dasd_diag.c +++ b/drivers/s390/block/dasd_diag.c | |||
| @@ -239,7 +239,6 @@ static void dasd_ext_handler(unsigned int ext_int_code, | |||
| 239 | addr_t ip; | 239 | addr_t ip; |
| 240 | int rc; | 240 | int rc; |
| 241 | 241 | ||
| 242 | kstat_cpu(smp_processor_id()).irqs[EXTINT_DSD]++; | ||
| 243 | switch (ext_int_code >> 24) { | 242 | switch (ext_int_code >> 24) { |
| 244 | case DASD_DIAG_CODE_31BIT: | 243 | case DASD_DIAG_CODE_31BIT: |
| 245 | ip = (addr_t) param32; | 244 | ip = (addr_t) param32; |
| @@ -250,6 +249,7 @@ static void dasd_ext_handler(unsigned int ext_int_code, | |||
| 250 | default: | 249 | default: |
| 251 | return; | 250 | return; |
| 252 | } | 251 | } |
| 252 | kstat_cpu(smp_processor_id()).irqs[EXTINT_DSD]++; | ||
| 253 | if (!ip) { /* no intparm: unsolicited interrupt */ | 253 | if (!ip) { /* no intparm: unsolicited interrupt */ |
| 254 | DBF_EVENT(DBF_NOTICE, "%s", "caught unsolicited " | 254 | DBF_EVENT(DBF_NOTICE, "%s", "caught unsolicited " |
| 255 | "interrupt"); | 255 | "interrupt"); |
diff --git a/drivers/s390/kvm/kvm_virtio.c b/drivers/s390/kvm/kvm_virtio.c index 414427d64a8f..607998f0b7d8 100644 --- a/drivers/s390/kvm/kvm_virtio.c +++ b/drivers/s390/kvm/kvm_virtio.c | |||
| @@ -381,10 +381,10 @@ static void kvm_extint_handler(unsigned int ext_int_code, | |||
| 381 | u16 subcode; | 381 | u16 subcode; |
| 382 | u32 param; | 382 | u32 param; |
| 383 | 383 | ||
| 384 | kstat_cpu(smp_processor_id()).irqs[EXTINT_VRT]++; | ||
| 385 | subcode = ext_int_code >> 16; | 384 | subcode = ext_int_code >> 16; |
| 386 | if ((subcode & 0xff00) != VIRTIO_SUBCODE_64) | 385 | if ((subcode & 0xff00) != VIRTIO_SUBCODE_64) |
| 387 | return; | 386 | return; |
| 387 | kstat_cpu(smp_processor_id()).irqs[EXTINT_VRT]++; | ||
| 388 | 388 | ||
| 389 | /* The LSB might be overloaded, we have to mask it */ | 389 | /* The LSB might be overloaded, we have to mask it */ |
| 390 | vq = (struct virtqueue *)(param64 & ~1UL); | 390 | vq = (struct virtqueue *)(param64 & ~1UL); |
diff --git a/drivers/scsi/device_handler/scsi_dh.c b/drivers/scsi/device_handler/scsi_dh.c index 564e6ecd17c2..0119b8147797 100644 --- a/drivers/scsi/device_handler/scsi_dh.c +++ b/drivers/scsi/device_handler/scsi_dh.c | |||
| @@ -394,12 +394,14 @@ int scsi_dh_activate(struct request_queue *q, activate_complete fn, void *data) | |||
| 394 | unsigned long flags; | 394 | unsigned long flags; |
| 395 | struct scsi_device *sdev; | 395 | struct scsi_device *sdev; |
| 396 | struct scsi_device_handler *scsi_dh = NULL; | 396 | struct scsi_device_handler *scsi_dh = NULL; |
| 397 | struct device *dev = NULL; | ||
| 397 | 398 | ||
| 398 | spin_lock_irqsave(q->queue_lock, flags); | 399 | spin_lock_irqsave(q->queue_lock, flags); |
| 399 | sdev = q->queuedata; | 400 | sdev = q->queuedata; |
| 400 | if (sdev && sdev->scsi_dh_data) | 401 | if (sdev && sdev->scsi_dh_data) |
| 401 | scsi_dh = sdev->scsi_dh_data->scsi_dh; | 402 | scsi_dh = sdev->scsi_dh_data->scsi_dh; |
| 402 | if (!scsi_dh || !get_device(&sdev->sdev_gendev) || | 403 | dev = get_device(&sdev->sdev_gendev); |
| 404 | if (!scsi_dh || !dev || | ||
| 403 | sdev->sdev_state == SDEV_CANCEL || | 405 | sdev->sdev_state == SDEV_CANCEL || |
| 404 | sdev->sdev_state == SDEV_DEL) | 406 | sdev->sdev_state == SDEV_DEL) |
| 405 | err = SCSI_DH_NOSYS; | 407 | err = SCSI_DH_NOSYS; |
| @@ -410,12 +412,13 @@ int scsi_dh_activate(struct request_queue *q, activate_complete fn, void *data) | |||
| 410 | if (err) { | 412 | if (err) { |
| 411 | if (fn) | 413 | if (fn) |
| 412 | fn(data, err); | 414 | fn(data, err); |
| 413 | return err; | 415 | goto out; |
| 414 | } | 416 | } |
| 415 | 417 | ||
| 416 | if (scsi_dh->activate) | 418 | if (scsi_dh->activate) |
| 417 | err = scsi_dh->activate(sdev, fn, data); | 419 | err = scsi_dh->activate(sdev, fn, data); |
| 418 | put_device(&sdev->sdev_gendev); | 420 | out: |
| 421 | put_device(dev); | ||
| 419 | return err; | 422 | return err; |
| 420 | } | 423 | } |
| 421 | EXPORT_SYMBOL_GPL(scsi_dh_activate); | 424 | EXPORT_SYMBOL_GPL(scsi_dh_activate); |
diff --git a/drivers/scsi/mpt2sas/mpt2sas_ctl.c b/drivers/scsi/mpt2sas/mpt2sas_ctl.c index 1c6d2b405eef..d72f1f2b1392 100644 --- a/drivers/scsi/mpt2sas/mpt2sas_ctl.c +++ b/drivers/scsi/mpt2sas/mpt2sas_ctl.c | |||
| @@ -688,6 +688,13 @@ _ctl_do_mpt_command(struct MPT2SAS_ADAPTER *ioc, | |||
| 688 | goto out; | 688 | goto out; |
| 689 | } | 689 | } |
| 690 | 690 | ||
| 691 | /* Check for overflow and wraparound */ | ||
| 692 | if (karg.data_sge_offset * 4 > ioc->request_sz || | ||
| 693 | karg.data_sge_offset > (UINT_MAX / 4)) { | ||
| 694 | ret = -EINVAL; | ||
| 695 | goto out; | ||
| 696 | } | ||
| 697 | |||
| 691 | /* copy in request message frame from user */ | 698 | /* copy in request message frame from user */ |
| 692 | if (copy_from_user(mpi_request, mf, karg.data_sge_offset*4)) { | 699 | if (copy_from_user(mpi_request, mf, karg.data_sge_offset*4)) { |
| 693 | printk(KERN_ERR "failure at %s:%d/%s()!\n", __FILE__, __LINE__, | 700 | printk(KERN_ERR "failure at %s:%d/%s()!\n", __FILE__, __LINE__, |
| @@ -1963,7 +1970,7 @@ _ctl_diag_read_buffer(void __user *arg, enum block_state state) | |||
| 1963 | Mpi2DiagBufferPostReply_t *mpi_reply; | 1970 | Mpi2DiagBufferPostReply_t *mpi_reply; |
| 1964 | int rc, i; | 1971 | int rc, i; |
| 1965 | u8 buffer_type; | 1972 | u8 buffer_type; |
| 1966 | unsigned long timeleft; | 1973 | unsigned long timeleft, request_size, copy_size; |
| 1967 | u16 smid; | 1974 | u16 smid; |
| 1968 | u16 ioc_status; | 1975 | u16 ioc_status; |
| 1969 | u8 issue_reset = 0; | 1976 | u8 issue_reset = 0; |
| @@ -1999,6 +2006,8 @@ _ctl_diag_read_buffer(void __user *arg, enum block_state state) | |||
| 1999 | return -ENOMEM; | 2006 | return -ENOMEM; |
| 2000 | } | 2007 | } |
| 2001 | 2008 | ||
| 2009 | request_size = ioc->diag_buffer_sz[buffer_type]; | ||
| 2010 | |||
| 2002 | if ((karg.starting_offset % 4) || (karg.bytes_to_read % 4)) { | 2011 | if ((karg.starting_offset % 4) || (karg.bytes_to_read % 4)) { |
| 2003 | printk(MPT2SAS_ERR_FMT "%s: either the starting_offset " | 2012 | printk(MPT2SAS_ERR_FMT "%s: either the starting_offset " |
| 2004 | "or bytes_to_read are not 4 byte aligned\n", ioc->name, | 2013 | "or bytes_to_read are not 4 byte aligned\n", ioc->name, |
| @@ -2006,13 +2015,23 @@ _ctl_diag_read_buffer(void __user *arg, enum block_state state) | |||
| 2006 | return -EINVAL; | 2015 | return -EINVAL; |
| 2007 | } | 2016 | } |
| 2008 | 2017 | ||
| 2018 | if (karg.starting_offset > request_size) | ||
| 2019 | return -EINVAL; | ||
| 2020 | |||
| 2009 | diag_data = (void *)(request_data + karg.starting_offset); | 2021 | diag_data = (void *)(request_data + karg.starting_offset); |
| 2010 | dctlprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: diag_buffer(%p), " | 2022 | dctlprintk(ioc, printk(MPT2SAS_INFO_FMT "%s: diag_buffer(%p), " |
| 2011 | "offset(%d), sz(%d)\n", ioc->name, __func__, | 2023 | "offset(%d), sz(%d)\n", ioc->name, __func__, |
| 2012 | diag_data, karg.starting_offset, karg.bytes_to_read)); | 2024 | diag_data, karg.starting_offset, karg.bytes_to_read)); |
| 2013 | 2025 | ||
| 2026 | /* Truncate data on requests that are too large */ | ||
| 2027 | if ((diag_data + karg.bytes_to_read < diag_data) || | ||
| 2028 | (diag_data + karg.bytes_to_read > request_data + request_size)) | ||
| 2029 | copy_size = request_size - karg.starting_offset; | ||
| 2030 | else | ||
| 2031 | copy_size = karg.bytes_to_read; | ||
| 2032 | |||
| 2014 | if (copy_to_user((void __user *)uarg->diagnostic_data, | 2033 | if (copy_to_user((void __user *)uarg->diagnostic_data, |
| 2015 | diag_data, karg.bytes_to_read)) { | 2034 | diag_data, copy_size)) { |
| 2016 | printk(MPT2SAS_ERR_FMT "%s: Unable to write " | 2035 | printk(MPT2SAS_ERR_FMT "%s: Unable to write " |
| 2017 | "mpt_diag_read_buffer_t data @ %p\n", ioc->name, | 2036 | "mpt_diag_read_buffer_t data @ %p\n", ioc->name, |
| 2018 | __func__, diag_data); | 2037 | __func__, diag_data); |
diff --git a/drivers/scsi/pmcraid.c b/drivers/scsi/pmcraid.c index 96d5ad0c1e42..7f636b118287 100644 --- a/drivers/scsi/pmcraid.c +++ b/drivers/scsi/pmcraid.c | |||
| @@ -3814,6 +3814,9 @@ static long pmcraid_ioctl_passthrough( | |||
| 3814 | rc = -EFAULT; | 3814 | rc = -EFAULT; |
| 3815 | goto out_free_buffer; | 3815 | goto out_free_buffer; |
| 3816 | } | 3816 | } |
| 3817 | } else if (request_size < 0) { | ||
| 3818 | rc = -EINVAL; | ||
| 3819 | goto out_free_buffer; | ||
| 3817 | } | 3820 | } |
| 3818 | 3821 | ||
| 3819 | /* check if we have any additional command parameters */ | 3822 | /* check if we have any additional command parameters */ |
diff --git a/drivers/scsi/scsi_sysfs.c b/drivers/scsi/scsi_sysfs.c index e44ff64233fd..e63912510fb9 100644 --- a/drivers/scsi/scsi_sysfs.c +++ b/drivers/scsi/scsi_sysfs.c | |||
| @@ -322,14 +322,8 @@ static void scsi_device_dev_release_usercontext(struct work_struct *work) | |||
| 322 | kfree(evt); | 322 | kfree(evt); |
| 323 | } | 323 | } |
| 324 | 324 | ||
| 325 | if (sdev->request_queue) { | 325 | /* NULL queue means the device can't be used */ |
| 326 | sdev->request_queue->queuedata = NULL; | 326 | sdev->request_queue = NULL; |
| 327 | /* user context needed to free queue */ | ||
| 328 | scsi_free_queue(sdev->request_queue); | ||
| 329 | /* temporary expedient, try to catch use of queue lock | ||
| 330 | * after free of sdev */ | ||
| 331 | sdev->request_queue = NULL; | ||
| 332 | } | ||
| 333 | 327 | ||
| 334 | scsi_target_reap(scsi_target(sdev)); | 328 | scsi_target_reap(scsi_target(sdev)); |
| 335 | 329 | ||
| @@ -937,6 +931,12 @@ void __scsi_remove_device(struct scsi_device *sdev) | |||
| 937 | if (sdev->host->hostt->slave_destroy) | 931 | if (sdev->host->hostt->slave_destroy) |
| 938 | sdev->host->hostt->slave_destroy(sdev); | 932 | sdev->host->hostt->slave_destroy(sdev); |
| 939 | transport_destroy_device(dev); | 933 | transport_destroy_device(dev); |
| 934 | |||
| 935 | /* cause the request function to reject all I/O requests */ | ||
| 936 | sdev->request_queue->queuedata = NULL; | ||
| 937 | |||
| 938 | /* Freeing the queue signals to block that we're done */ | ||
| 939 | scsi_free_queue(sdev->request_queue); | ||
| 940 | put_device(dev); | 940 | put_device(dev); |
| 941 | } | 941 | } |
| 942 | 942 | ||
diff --git a/drivers/staging/rt2860/common/cmm_data_pci.c b/drivers/staging/rt2860/common/cmm_data_pci.c index bef0bbd8cef7..f01a51c381f1 100644 --- a/drivers/staging/rt2860/common/cmm_data_pci.c +++ b/drivers/staging/rt2860/common/cmm_data_pci.c | |||
| @@ -444,7 +444,7 @@ int RTMPCheckRxError(struct rt_rtmp_adapter *pAd, | |||
| 444 | return (NDIS_STATUS_FAILURE); | 444 | return (NDIS_STATUS_FAILURE); |
| 445 | } | 445 | } |
| 446 | } | 446 | } |
| 447 | /* Drop not U2M frames, can't's drop here because we will drop beacon in this case */ | 447 | /* Drop not U2M frames, can't drop here because we will drop beacon in this case */ |
| 448 | /* I am kind of doubting the U2M bit operation */ | 448 | /* I am kind of doubting the U2M bit operation */ |
| 449 | /* if (pRxD->U2M == 0) */ | 449 | /* if (pRxD->U2M == 0) */ |
| 450 | /* return(NDIS_STATUS_FAILURE); */ | 450 | /* return(NDIS_STATUS_FAILURE); */ |
diff --git a/drivers/staging/rt2860/common/cmm_data_usb.c b/drivers/staging/rt2860/common/cmm_data_usb.c index 5637857ae9eb..83a62faa7e57 100644 --- a/drivers/staging/rt2860/common/cmm_data_usb.c +++ b/drivers/staging/rt2860/common/cmm_data_usb.c | |||
| @@ -860,7 +860,7 @@ int RTMPCheckRxError(struct rt_rtmp_adapter *pAd, | |||
| 860 | DBGPRINT_RAW(RT_DEBUG_ERROR, ("received packet too long\n")); | 860 | DBGPRINT_RAW(RT_DEBUG_ERROR, ("received packet too long\n")); |
| 861 | return NDIS_STATUS_FAILURE; | 861 | return NDIS_STATUS_FAILURE; |
| 862 | } | 862 | } |
| 863 | /* Drop not U2M frames, can't's drop here because we will drop beacon in this case */ | 863 | /* Drop not U2M frames, can't drop here because we will drop beacon in this case */ |
| 864 | /* I am kind of doubting the U2M bit operation */ | 864 | /* I am kind of doubting the U2M bit operation */ |
| 865 | /* if (pRxD->U2M == 0) */ | 865 | /* if (pRxD->U2M == 0) */ |
| 866 | /* return(NDIS_STATUS_FAILURE); */ | 866 | /* return(NDIS_STATUS_FAILURE); */ |
diff --git a/drivers/staging/spectra/ffsport.c b/drivers/staging/spectra/ffsport.c index 20dae73d3b78..506547b603e1 100644 --- a/drivers/staging/spectra/ffsport.c +++ b/drivers/staging/spectra/ffsport.c | |||
| @@ -653,7 +653,7 @@ static int SBD_setup_device(struct spectra_nand_dev *dev, int which) | |||
| 653 | } | 653 | } |
| 654 | dev->queue->queuedata = dev; | 654 | dev->queue->queuedata = dev; |
| 655 | 655 | ||
| 656 | /* As Linux block layer does't support >4KB hardware sector, */ | 656 | /* As Linux block layer doesn't support >4KB hardware sector, */ |
| 657 | /* Here we force report 512 byte hardware sector size to Kernel */ | 657 | /* Here we force report 512 byte hardware sector size to Kernel */ |
| 658 | blk_queue_logical_block_size(dev->queue, 512); | 658 | blk_queue_logical_block_size(dev->queue, 512); |
| 659 | 659 | ||
diff --git a/drivers/staging/tidspbridge/dynload/cload.c b/drivers/staging/tidspbridge/dynload/cload.c index 5cecd237e3f6..fe1ef0addb09 100644 --- a/drivers/staging/tidspbridge/dynload/cload.c +++ b/drivers/staging/tidspbridge/dynload/cload.c | |||
| @@ -718,7 +718,7 @@ static void dload_symbols(struct dload_state *dlthis) | |||
| 718 | * as a temporary for .dllview record construction. | 718 | * as a temporary for .dllview record construction. |
| 719 | * Allocate storage for the whole table. Add 1 to the section count | 719 | * Allocate storage for the whole table. Add 1 to the section count |
| 720 | * in case a trampoline section is auto-generated as well as the | 720 | * in case a trampoline section is auto-generated as well as the |
| 721 | * size of the trampoline section name so DLLView does't get lost. | 721 | * size of the trampoline section name so DLLView doesn't get lost. |
| 722 | */ | 722 | */ |
| 723 | 723 | ||
| 724 | siz = sym_count * sizeof(struct local_symbol); | 724 | siz = sym_count * sizeof(struct local_symbol); |
diff --git a/drivers/staging/tty/specialix.c b/drivers/staging/tty/specialix.c index cb24c6d999db..5c3598ec7456 100644 --- a/drivers/staging/tty/specialix.c +++ b/drivers/staging/tty/specialix.c | |||
| @@ -978,7 +978,7 @@ static void sx_change_speed(struct specialix_board *bp, | |||
| 978 | spin_lock_irqsave(&bp->lock, flags); | 978 | spin_lock_irqsave(&bp->lock, flags); |
| 979 | sx_out(bp, CD186x_CAR, port_No(port)); | 979 | sx_out(bp, CD186x_CAR, port_No(port)); |
| 980 | 980 | ||
| 981 | /* The Specialix board does't implement the RTS lines. | 981 | /* The Specialix board doesn't implement the RTS lines. |
| 982 | They are used to set the IRQ level. Don't touch them. */ | 982 | They are used to set the IRQ level. Don't touch them. */ |
| 983 | if (sx_crtscts(tty)) | 983 | if (sx_crtscts(tty)) |
| 984 | port->MSVR = MSVR_DTR | (sx_in(bp, CD186x_MSVR) & MSVR_RTS); | 984 | port->MSVR = MSVR_DTR | (sx_in(bp, CD186x_MSVR) & MSVR_RTS); |
diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h index 2e61fe1b6b8c..8f4b81de3ae2 100644 --- a/fs/btrfs/ctree.h +++ b/fs/btrfs/ctree.h | |||
| @@ -718,7 +718,7 @@ struct btrfs_space_info { | |||
| 718 | u64 total_bytes; /* total bytes in the space, | 718 | u64 total_bytes; /* total bytes in the space, |
| 719 | this doesn't take mirrors into account */ | 719 | this doesn't take mirrors into account */ |
| 720 | u64 bytes_used; /* total bytes used, | 720 | u64 bytes_used; /* total bytes used, |
| 721 | this does't take mirrors into account */ | 721 | this doesn't take mirrors into account */ |
| 722 | u64 bytes_pinned; /* total bytes pinned, will be freed when the | 722 | u64 bytes_pinned; /* total bytes pinned, will be freed when the |
| 723 | transaction finishes */ | 723 | transaction finishes */ |
| 724 | u64 bytes_reserved; /* total bytes the allocator has reserved for | 724 | u64 bytes_reserved; /* total bytes the allocator has reserved for |
| @@ -9,6 +9,7 @@ | |||
| 9 | #include <linux/module.h> | 9 | #include <linux/module.h> |
| 10 | #include <linux/fs.h> | 10 | #include <linux/fs.h> |
| 11 | #include <linux/mm.h> | 11 | #include <linux/mm.h> |
| 12 | #include <linux/mmzone.h> | ||
| 12 | #include <linux/time.h> | 13 | #include <linux/time.h> |
| 13 | #include <linux/sched.h> | 14 | #include <linux/sched.h> |
| 14 | #include <linux/slab.h> | 15 | #include <linux/slab.h> |
| @@ -39,14 +40,17 @@ int sysctl_nr_open_max = 1024 * 1024; /* raised later */ | |||
| 39 | */ | 40 | */ |
| 40 | static DEFINE_PER_CPU(struct fdtable_defer, fdtable_defer_list); | 41 | static DEFINE_PER_CPU(struct fdtable_defer, fdtable_defer_list); |
| 41 | 42 | ||
| 42 | static inline void *alloc_fdmem(unsigned int size) | 43 | static void *alloc_fdmem(unsigned int size) |
| 43 | { | 44 | { |
| 44 | void *data; | 45 | /* |
| 45 | 46 | * Very large allocations can stress page reclaim, so fall back to | |
| 46 | data = kmalloc(size, GFP_KERNEL|__GFP_NOWARN); | 47 | * vmalloc() if the allocation size will be considered "large" by the VM. |
| 47 | if (data != NULL) | 48 | */ |
| 48 | return data; | 49 | if (size <= (PAGE_SIZE << PAGE_ALLOC_COSTLY_ORDER)) { |
| 49 | 50 | void *data = kmalloc(size, GFP_KERNEL|__GFP_NOWARN); | |
| 51 | if (data != NULL) | ||
| 52 | return data; | ||
| 53 | } | ||
| 50 | return vmalloc(size); | 54 | return vmalloc(size); |
| 51 | } | 55 | } |
| 52 | 56 | ||
diff --git a/fs/logfs/super.c b/fs/logfs/super.c index 33435e4b14d2..ce03a182c771 100644 --- a/fs/logfs/super.c +++ b/fs/logfs/super.c | |||
| @@ -480,10 +480,6 @@ static int logfs_read_sb(struct super_block *sb, int read_only) | |||
| 480 | !read_only) | 480 | !read_only) |
| 481 | return -EIO; | 481 | return -EIO; |
| 482 | 482 | ||
| 483 | mutex_init(&super->s_dirop_mutex); | ||
| 484 | mutex_init(&super->s_object_alias_mutex); | ||
| 485 | INIT_LIST_HEAD(&super->s_freeing_list); | ||
| 486 | |||
| 487 | ret = logfs_init_rw(sb); | 483 | ret = logfs_init_rw(sb); |
| 488 | if (ret) | 484 | if (ret) |
| 489 | return ret; | 485 | return ret; |
| @@ -601,6 +597,10 @@ static struct dentry *logfs_mount(struct file_system_type *type, int flags, | |||
| 601 | if (!super) | 597 | if (!super) |
| 602 | return ERR_PTR(-ENOMEM); | 598 | return ERR_PTR(-ENOMEM); |
| 603 | 599 | ||
| 600 | mutex_init(&super->s_dirop_mutex); | ||
| 601 | mutex_init(&super->s_object_alias_mutex); | ||
| 602 | INIT_LIST_HEAD(&super->s_freeing_list); | ||
| 603 | |||
| 604 | if (!devname) | 604 | if (!devname) |
| 605 | err = logfs_get_sb_bdev(super, type, devname); | 605 | err = logfs_get_sb_bdev(super, type, devname); |
| 606 | else if (strncmp(devname, "mtd", 3)) | 606 | else if (strncmp(devname, "mtd", 3)) |
diff --git a/fs/nfs/namespace.c b/fs/nfs/namespace.c index 89fc160fd5b0..1f063bacd285 100644 --- a/fs/nfs/namespace.c +++ b/fs/nfs/namespace.c | |||
| @@ -119,7 +119,7 @@ Elong: | |||
| 119 | } | 119 | } |
| 120 | 120 | ||
| 121 | #ifdef CONFIG_NFS_V4 | 121 | #ifdef CONFIG_NFS_V4 |
| 122 | static rpc_authflavor_t nfs_find_best_sec(struct nfs4_secinfo_flavors *flavors, struct inode *inode) | 122 | static rpc_authflavor_t nfs_find_best_sec(struct nfs4_secinfo_flavors *flavors) |
| 123 | { | 123 | { |
| 124 | struct gss_api_mech *mech; | 124 | struct gss_api_mech *mech; |
| 125 | struct xdr_netobj oid; | 125 | struct xdr_netobj oid; |
| @@ -166,7 +166,7 @@ static int nfs_negotiate_security(const struct dentry *parent, | |||
| 166 | } | 166 | } |
| 167 | flavors = page_address(page); | 167 | flavors = page_address(page); |
| 168 | ret = secinfo(parent->d_inode, &dentry->d_name, flavors); | 168 | ret = secinfo(parent->d_inode, &dentry->d_name, flavors); |
| 169 | *flavor = nfs_find_best_sec(flavors, dentry->d_inode); | 169 | *flavor = nfs_find_best_sec(flavors); |
| 170 | put_page(page); | 170 | put_page(page); |
| 171 | } | 171 | } |
| 172 | 172 | ||
diff --git a/fs/nfs/nfs4_fs.h b/fs/nfs/nfs4_fs.h index e1c261ddd65d..c4a69833dd0d 100644 --- a/fs/nfs/nfs4_fs.h +++ b/fs/nfs/nfs4_fs.h | |||
| @@ -47,6 +47,7 @@ enum nfs4_client_state { | |||
| 47 | NFS4CLNT_LAYOUTRECALL, | 47 | NFS4CLNT_LAYOUTRECALL, |
| 48 | NFS4CLNT_SESSION_RESET, | 48 | NFS4CLNT_SESSION_RESET, |
| 49 | NFS4CLNT_RECALL_SLOT, | 49 | NFS4CLNT_RECALL_SLOT, |
| 50 | NFS4CLNT_LEASE_CONFIRM, | ||
| 50 | }; | 51 | }; |
| 51 | 52 | ||
| 52 | enum nfs4_session_state { | 53 | enum nfs4_session_state { |
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c index 9bf41eab3e46..69c0f3c5ee7a 100644 --- a/fs/nfs/nfs4proc.c +++ b/fs/nfs/nfs4proc.c | |||
| @@ -46,6 +46,7 @@ | |||
| 46 | #include <linux/nfs4.h> | 46 | #include <linux/nfs4.h> |
| 47 | #include <linux/nfs_fs.h> | 47 | #include <linux/nfs_fs.h> |
| 48 | #include <linux/nfs_page.h> | 48 | #include <linux/nfs_page.h> |
| 49 | #include <linux/nfs_mount.h> | ||
| 49 | #include <linux/namei.h> | 50 | #include <linux/namei.h> |
| 50 | #include <linux/mount.h> | 51 | #include <linux/mount.h> |
| 51 | #include <linux/module.h> | 52 | #include <linux/module.h> |
| @@ -443,8 +444,8 @@ static int nfs41_sequence_done(struct rpc_task *task, struct nfs4_sequence_res * | |||
| 443 | if (res->sr_status == 1) | 444 | if (res->sr_status == 1) |
| 444 | res->sr_status = NFS_OK; | 445 | res->sr_status = NFS_OK; |
| 445 | 446 | ||
| 446 | /* -ERESTARTSYS can result in skipping nfs41_sequence_setup */ | 447 | /* don't increment the sequence number if the task wasn't sent */ |
| 447 | if (!res->sr_slot) | 448 | if (!RPC_WAS_SENT(task)) |
| 448 | goto out; | 449 | goto out; |
| 449 | 450 | ||
| 450 | /* Check the SEQUENCE operation status */ | 451 | /* Check the SEQUENCE operation status */ |
| @@ -2185,9 +2186,14 @@ static int nfs4_lookup_root(struct nfs_server *server, struct nfs_fh *fhandle, | |||
| 2185 | struct nfs4_exception exception = { }; | 2186 | struct nfs4_exception exception = { }; |
| 2186 | int err; | 2187 | int err; |
| 2187 | do { | 2188 | do { |
| 2188 | err = nfs4_handle_exception(server, | 2189 | err = _nfs4_lookup_root(server, fhandle, info); |
| 2189 | _nfs4_lookup_root(server, fhandle, info), | 2190 | switch (err) { |
| 2190 | &exception); | 2191 | case 0: |
| 2192 | case -NFS4ERR_WRONGSEC: | ||
| 2193 | break; | ||
| 2194 | default: | ||
| 2195 | err = nfs4_handle_exception(server, err, &exception); | ||
| 2196 | } | ||
| 2191 | } while (exception.retry); | 2197 | } while (exception.retry); |
| 2192 | return err; | 2198 | return err; |
| 2193 | } | 2199 | } |
| @@ -2208,25 +2214,47 @@ out: | |||
| 2208 | return ret; | 2214 | return ret; |
| 2209 | } | 2215 | } |
| 2210 | 2216 | ||
| 2211 | /* | 2217 | static int nfs4_find_root_sec(struct nfs_server *server, struct nfs_fh *fhandle, |
| 2212 | * get the file handle for the "/" directory on the server | ||
| 2213 | */ | ||
| 2214 | static int nfs4_proc_get_root(struct nfs_server *server, struct nfs_fh *fhandle, | ||
| 2215 | struct nfs_fsinfo *info) | 2218 | struct nfs_fsinfo *info) |
| 2216 | { | 2219 | { |
| 2217 | int i, len, status = 0; | 2220 | int i, len, status = 0; |
| 2218 | rpc_authflavor_t flav_array[NFS_MAX_SECFLAVORS + 2]; | 2221 | rpc_authflavor_t flav_array[NFS_MAX_SECFLAVORS]; |
| 2219 | 2222 | ||
| 2220 | flav_array[0] = RPC_AUTH_UNIX; | 2223 | len = gss_mech_list_pseudoflavors(&flav_array[0]); |
| 2221 | len = gss_mech_list_pseudoflavors(&flav_array[1]); | 2224 | flav_array[len] = RPC_AUTH_NULL; |
| 2222 | flav_array[1+len] = RPC_AUTH_NULL; | 2225 | len += 1; |
| 2223 | len += 2; | ||
| 2224 | 2226 | ||
| 2225 | for (i = 0; i < len; i++) { | 2227 | for (i = 0; i < len; i++) { |
| 2226 | status = nfs4_lookup_root_sec(server, fhandle, info, flav_array[i]); | 2228 | status = nfs4_lookup_root_sec(server, fhandle, info, flav_array[i]); |
| 2227 | if (status != -EPERM) | 2229 | if (status == -NFS4ERR_WRONGSEC || status == -EACCES) |
| 2228 | break; | 2230 | continue; |
| 2231 | break; | ||
| 2229 | } | 2232 | } |
| 2233 | /* | ||
| 2234 | * -EACCESS could mean that the user doesn't have correct permissions | ||
| 2235 | * to access the mount. It could also mean that we tried to mount | ||
| 2236 | * with a gss auth flavor, but rpc.gssd isn't running. Either way, | ||
| 2237 | * existing mount programs don't handle -EACCES very well so it should | ||
| 2238 | * be mapped to -EPERM instead. | ||
| 2239 | */ | ||
| 2240 | if (status == -EACCES) | ||
| 2241 | status = -EPERM; | ||
| 2242 | return status; | ||
| 2243 | } | ||
| 2244 | |||
| 2245 | /* | ||
| 2246 | * get the file handle for the "/" directory on the server | ||
| 2247 | */ | ||
| 2248 | static int nfs4_proc_get_root(struct nfs_server *server, struct nfs_fh *fhandle, | ||
| 2249 | struct nfs_fsinfo *info) | ||
| 2250 | { | ||
| 2251 | int status = nfs4_lookup_root(server, fhandle, info); | ||
| 2252 | if ((status == -NFS4ERR_WRONGSEC) && !(server->flags & NFS_MOUNT_SECFLAVOUR)) | ||
| 2253 | /* | ||
| 2254 | * A status of -NFS4ERR_WRONGSEC will be mapped to -EPERM | ||
| 2255 | * by nfs4_map_errors() as this function exits. | ||
| 2256 | */ | ||
| 2257 | status = nfs4_find_root_sec(server, fhandle, info); | ||
| 2230 | if (status == 0) | 2258 | if (status == 0) |
| 2231 | status = nfs4_server_capabilities(server, fhandle); | 2259 | status = nfs4_server_capabilities(server, fhandle); |
| 2232 | if (status == 0) | 2260 | if (status == 0) |
| @@ -3723,21 +3751,20 @@ int nfs4_proc_setclientid(struct nfs_client *clp, u32 program, | |||
| 3723 | sizeof(setclientid.sc_uaddr), "%s.%u.%u", | 3751 | sizeof(setclientid.sc_uaddr), "%s.%u.%u", |
| 3724 | clp->cl_ipaddr, port >> 8, port & 255); | 3752 | clp->cl_ipaddr, port >> 8, port & 255); |
| 3725 | 3753 | ||
| 3726 | status = rpc_call_sync(clp->cl_rpcclient, &msg, 0); | 3754 | status = rpc_call_sync(clp->cl_rpcclient, &msg, RPC_TASK_TIMEOUT); |
| 3727 | if (status != -NFS4ERR_CLID_INUSE) | 3755 | if (status != -NFS4ERR_CLID_INUSE) |
| 3728 | break; | 3756 | break; |
| 3729 | if (signalled()) | 3757 | if (loop != 0) { |
| 3758 | ++clp->cl_id_uniquifier; | ||
| 3730 | break; | 3759 | break; |
| 3731 | if (loop++ & 1) | 3760 | } |
| 3732 | ssleep(clp->cl_lease_time / HZ + 1); | 3761 | ++loop; |
| 3733 | else | 3762 | ssleep(clp->cl_lease_time / HZ + 1); |
| 3734 | if (++clp->cl_id_uniquifier == 0) | ||
| 3735 | break; | ||
| 3736 | } | 3763 | } |
| 3737 | return status; | 3764 | return status; |
| 3738 | } | 3765 | } |
| 3739 | 3766 | ||
| 3740 | static int _nfs4_proc_setclientid_confirm(struct nfs_client *clp, | 3767 | int nfs4_proc_setclientid_confirm(struct nfs_client *clp, |
| 3741 | struct nfs4_setclientid_res *arg, | 3768 | struct nfs4_setclientid_res *arg, |
| 3742 | struct rpc_cred *cred) | 3769 | struct rpc_cred *cred) |
| 3743 | { | 3770 | { |
| @@ -3752,7 +3779,7 @@ static int _nfs4_proc_setclientid_confirm(struct nfs_client *clp, | |||
| 3752 | int status; | 3779 | int status; |
| 3753 | 3780 | ||
| 3754 | now = jiffies; | 3781 | now = jiffies; |
| 3755 | status = rpc_call_sync(clp->cl_rpcclient, &msg, 0); | 3782 | status = rpc_call_sync(clp->cl_rpcclient, &msg, RPC_TASK_TIMEOUT); |
| 3756 | if (status == 0) { | 3783 | if (status == 0) { |
| 3757 | spin_lock(&clp->cl_lock); | 3784 | spin_lock(&clp->cl_lock); |
| 3758 | clp->cl_lease_time = fsinfo.lease_time * HZ; | 3785 | clp->cl_lease_time = fsinfo.lease_time * HZ; |
| @@ -3762,26 +3789,6 @@ static int _nfs4_proc_setclientid_confirm(struct nfs_client *clp, | |||
| 3762 | return status; | 3789 | return status; |
| 3763 | } | 3790 | } |
| 3764 | 3791 | ||
| 3765 | int nfs4_proc_setclientid_confirm(struct nfs_client *clp, | ||
| 3766 | struct nfs4_setclientid_res *arg, | ||
| 3767 | struct rpc_cred *cred) | ||
| 3768 | { | ||
| 3769 | long timeout = 0; | ||
| 3770 | int err; | ||
| 3771 | do { | ||
| 3772 | err = _nfs4_proc_setclientid_confirm(clp, arg, cred); | ||
| 3773 | switch (err) { | ||
| 3774 | case 0: | ||
| 3775 | return err; | ||
| 3776 | case -NFS4ERR_RESOURCE: | ||
| 3777 | /* The IBM lawyers misread another document! */ | ||
| 3778 | case -NFS4ERR_DELAY: | ||
| 3779 | err = nfs4_delay(clp->cl_rpcclient, &timeout); | ||
| 3780 | } | ||
| 3781 | } while (err == 0); | ||
| 3782 | return err; | ||
| 3783 | } | ||
| 3784 | |||
| 3785 | struct nfs4_delegreturndata { | 3792 | struct nfs4_delegreturndata { |
| 3786 | struct nfs4_delegreturnargs args; | 3793 | struct nfs4_delegreturnargs args; |
| 3787 | struct nfs4_delegreturnres res; | 3794 | struct nfs4_delegreturnres res; |
| @@ -4786,7 +4793,7 @@ int nfs4_proc_exchange_id(struct nfs_client *clp, struct rpc_cred *cred) | |||
| 4786 | init_utsname()->domainname, | 4793 | init_utsname()->domainname, |
| 4787 | clp->cl_rpcclient->cl_auth->au_flavor); | 4794 | clp->cl_rpcclient->cl_auth->au_flavor); |
| 4788 | 4795 | ||
| 4789 | status = rpc_call_sync(clp->cl_rpcclient, &msg, 0); | 4796 | status = rpc_call_sync(clp->cl_rpcclient, &msg, RPC_TASK_TIMEOUT); |
| 4790 | if (!status) | 4797 | if (!status) |
| 4791 | status = nfs4_check_cl_exchange_flags(clp->cl_exchange_flags); | 4798 | status = nfs4_check_cl_exchange_flags(clp->cl_exchange_flags); |
| 4792 | dprintk("<-- %s status= %d\n", __func__, status); | 4799 | dprintk("<-- %s status= %d\n", __func__, status); |
| @@ -4869,7 +4876,8 @@ int nfs4_proc_get_lease_time(struct nfs_client *clp, struct nfs_fsinfo *fsinfo) | |||
| 4869 | .rpc_client = clp->cl_rpcclient, | 4876 | .rpc_client = clp->cl_rpcclient, |
| 4870 | .rpc_message = &msg, | 4877 | .rpc_message = &msg, |
| 4871 | .callback_ops = &nfs4_get_lease_time_ops, | 4878 | .callback_ops = &nfs4_get_lease_time_ops, |
| 4872 | .callback_data = &data | 4879 | .callback_data = &data, |
| 4880 | .flags = RPC_TASK_TIMEOUT, | ||
| 4873 | }; | 4881 | }; |
| 4874 | int status; | 4882 | int status; |
| 4875 | 4883 | ||
| @@ -5171,7 +5179,7 @@ static int _nfs4_proc_create_session(struct nfs_client *clp) | |||
| 5171 | nfs4_init_channel_attrs(&args); | 5179 | nfs4_init_channel_attrs(&args); |
| 5172 | args.flags = (SESSION4_PERSIST | SESSION4_BACK_CHAN); | 5180 | args.flags = (SESSION4_PERSIST | SESSION4_BACK_CHAN); |
| 5173 | 5181 | ||
| 5174 | status = rpc_call_sync(session->clp->cl_rpcclient, &msg, 0); | 5182 | status = rpc_call_sync(session->clp->cl_rpcclient, &msg, RPC_TASK_TIMEOUT); |
| 5175 | 5183 | ||
| 5176 | if (!status) | 5184 | if (!status) |
| 5177 | /* Verify the session's negotiated channel_attrs values */ | 5185 | /* Verify the session's negotiated channel_attrs values */ |
| @@ -5194,20 +5202,10 @@ int nfs4_proc_create_session(struct nfs_client *clp) | |||
| 5194 | int status; | 5202 | int status; |
| 5195 | unsigned *ptr; | 5203 | unsigned *ptr; |
| 5196 | struct nfs4_session *session = clp->cl_session; | 5204 | struct nfs4_session *session = clp->cl_session; |
| 5197 | long timeout = 0; | ||
| 5198 | int err; | ||
| 5199 | 5205 | ||
| 5200 | dprintk("--> %s clp=%p session=%p\n", __func__, clp, session); | 5206 | dprintk("--> %s clp=%p session=%p\n", __func__, clp, session); |
| 5201 | 5207 | ||
| 5202 | do { | 5208 | status = _nfs4_proc_create_session(clp); |
| 5203 | status = _nfs4_proc_create_session(clp); | ||
| 5204 | if (status == -NFS4ERR_DELAY) { | ||
| 5205 | err = nfs4_delay(clp->cl_rpcclient, &timeout); | ||
| 5206 | if (err) | ||
| 5207 | status = err; | ||
| 5208 | } | ||
| 5209 | } while (status == -NFS4ERR_DELAY); | ||
| 5210 | |||
| 5211 | if (status) | 5209 | if (status) |
| 5212 | goto out; | 5210 | goto out; |
| 5213 | 5211 | ||
| @@ -5248,7 +5246,7 @@ int nfs4_proc_destroy_session(struct nfs4_session *session) | |||
| 5248 | msg.rpc_argp = session; | 5246 | msg.rpc_argp = session; |
| 5249 | msg.rpc_resp = NULL; | 5247 | msg.rpc_resp = NULL; |
| 5250 | msg.rpc_cred = NULL; | 5248 | msg.rpc_cred = NULL; |
| 5251 | status = rpc_call_sync(session->clp->cl_rpcclient, &msg, 0); | 5249 | status = rpc_call_sync(session->clp->cl_rpcclient, &msg, RPC_TASK_TIMEOUT); |
| 5252 | 5250 | ||
| 5253 | if (status) | 5251 | if (status) |
| 5254 | printk(KERN_WARNING | 5252 | printk(KERN_WARNING |
diff --git a/fs/nfs/nfs4state.c b/fs/nfs/nfs4state.c index a6804f704d9d..036f5adc9e1f 100644 --- a/fs/nfs/nfs4state.c +++ b/fs/nfs/nfs4state.c | |||
| @@ -64,10 +64,15 @@ static LIST_HEAD(nfs4_clientid_list); | |||
| 64 | 64 | ||
| 65 | int nfs4_init_clientid(struct nfs_client *clp, struct rpc_cred *cred) | 65 | int nfs4_init_clientid(struct nfs_client *clp, struct rpc_cred *cred) |
| 66 | { | 66 | { |
| 67 | struct nfs4_setclientid_res clid; | 67 | struct nfs4_setclientid_res clid = { |
| 68 | .clientid = clp->cl_clientid, | ||
| 69 | .confirm = clp->cl_confirm, | ||
| 70 | }; | ||
| 68 | unsigned short port; | 71 | unsigned short port; |
| 69 | int status; | 72 | int status; |
| 70 | 73 | ||
| 74 | if (test_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state)) | ||
| 75 | goto do_confirm; | ||
| 71 | port = nfs_callback_tcpport; | 76 | port = nfs_callback_tcpport; |
| 72 | if (clp->cl_addr.ss_family == AF_INET6) | 77 | if (clp->cl_addr.ss_family == AF_INET6) |
| 73 | port = nfs_callback_tcpport6; | 78 | port = nfs_callback_tcpport6; |
| @@ -75,10 +80,14 @@ int nfs4_init_clientid(struct nfs_client *clp, struct rpc_cred *cred) | |||
| 75 | status = nfs4_proc_setclientid(clp, NFS4_CALLBACK, port, cred, &clid); | 80 | status = nfs4_proc_setclientid(clp, NFS4_CALLBACK, port, cred, &clid); |
| 76 | if (status != 0) | 81 | if (status != 0) |
| 77 | goto out; | 82 | goto out; |
| 83 | clp->cl_clientid = clid.clientid; | ||
| 84 | clp->cl_confirm = clid.confirm; | ||
| 85 | set_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state); | ||
| 86 | do_confirm: | ||
| 78 | status = nfs4_proc_setclientid_confirm(clp, &clid, cred); | 87 | status = nfs4_proc_setclientid_confirm(clp, &clid, cred); |
| 79 | if (status != 0) | 88 | if (status != 0) |
| 80 | goto out; | 89 | goto out; |
| 81 | clp->cl_clientid = clid.clientid; | 90 | clear_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state); |
| 82 | nfs4_schedule_state_renewal(clp); | 91 | nfs4_schedule_state_renewal(clp); |
| 83 | out: | 92 | out: |
| 84 | return status; | 93 | return status; |
| @@ -230,13 +239,18 @@ int nfs41_init_clientid(struct nfs_client *clp, struct rpc_cred *cred) | |||
| 230 | { | 239 | { |
| 231 | int status; | 240 | int status; |
| 232 | 241 | ||
| 242 | if (test_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state)) | ||
| 243 | goto do_confirm; | ||
| 233 | nfs4_begin_drain_session(clp); | 244 | nfs4_begin_drain_session(clp); |
| 234 | status = nfs4_proc_exchange_id(clp, cred); | 245 | status = nfs4_proc_exchange_id(clp, cred); |
| 235 | if (status != 0) | 246 | if (status != 0) |
| 236 | goto out; | 247 | goto out; |
| 248 | set_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state); | ||
| 249 | do_confirm: | ||
| 237 | status = nfs4_proc_create_session(clp); | 250 | status = nfs4_proc_create_session(clp); |
| 238 | if (status != 0) | 251 | if (status != 0) |
| 239 | goto out; | 252 | goto out; |
| 253 | clear_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state); | ||
| 240 | nfs41_setup_state_renewal(clp); | 254 | nfs41_setup_state_renewal(clp); |
| 241 | nfs_mark_client_ready(clp, NFS_CS_READY); | 255 | nfs_mark_client_ready(clp, NFS_CS_READY); |
| 242 | out: | 256 | out: |
| @@ -1584,20 +1598,23 @@ static int nfs4_recall_slot(struct nfs_client *clp) { return 0; } | |||
| 1584 | */ | 1598 | */ |
| 1585 | static void nfs4_set_lease_expired(struct nfs_client *clp, int status) | 1599 | static void nfs4_set_lease_expired(struct nfs_client *clp, int status) |
| 1586 | { | 1600 | { |
| 1587 | if (nfs4_has_session(clp)) { | 1601 | switch (status) { |
| 1588 | switch (status) { | 1602 | case -NFS4ERR_CLID_INUSE: |
| 1589 | case -NFS4ERR_DELAY: | 1603 | case -NFS4ERR_STALE_CLIENTID: |
| 1590 | case -NFS4ERR_CLID_INUSE: | 1604 | clear_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state); |
| 1591 | case -EAGAIN: | 1605 | break; |
| 1592 | break; | 1606 | case -NFS4ERR_DELAY: |
| 1607 | case -ETIMEDOUT: | ||
| 1608 | case -EAGAIN: | ||
| 1609 | ssleep(1); | ||
| 1610 | break; | ||
| 1593 | 1611 | ||
| 1594 | case -EKEYEXPIRED: | 1612 | case -EKEYEXPIRED: |
| 1595 | nfs4_warn_keyexpired(clp->cl_hostname); | 1613 | nfs4_warn_keyexpired(clp->cl_hostname); |
| 1596 | case -NFS4ERR_NOT_SAME: /* FixMe: implement recovery | 1614 | case -NFS4ERR_NOT_SAME: /* FixMe: implement recovery |
| 1597 | * in nfs4_exchange_id */ | 1615 | * in nfs4_exchange_id */ |
| 1598 | default: | 1616 | default: |
| 1599 | return; | 1617 | return; |
| 1600 | } | ||
| 1601 | } | 1618 | } |
| 1602 | set_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state); | 1619 | set_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state); |
| 1603 | } | 1620 | } |
| @@ -1607,7 +1624,7 @@ static void nfs4_state_manager(struct nfs_client *clp) | |||
| 1607 | int status = 0; | 1624 | int status = 0; |
| 1608 | 1625 | ||
| 1609 | /* Ensure exclusive access to NFSv4 state */ | 1626 | /* Ensure exclusive access to NFSv4 state */ |
| 1610 | for(;;) { | 1627 | do { |
| 1611 | if (test_and_clear_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state)) { | 1628 | if (test_and_clear_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state)) { |
| 1612 | /* We're going to have to re-establish a clientid */ | 1629 | /* We're going to have to re-establish a clientid */ |
| 1613 | status = nfs4_reclaim_lease(clp); | 1630 | status = nfs4_reclaim_lease(clp); |
| @@ -1691,7 +1708,7 @@ static void nfs4_state_manager(struct nfs_client *clp) | |||
| 1691 | break; | 1708 | break; |
| 1692 | if (test_and_set_bit(NFS4CLNT_MANAGER_RUNNING, &clp->cl_state) != 0) | 1709 | if (test_and_set_bit(NFS4CLNT_MANAGER_RUNNING, &clp->cl_state) != 0) |
| 1693 | break; | 1710 | break; |
| 1694 | } | 1711 | } while (atomic_read(&clp->cl_count) > 1); |
| 1695 | return; | 1712 | return; |
| 1696 | out_error: | 1713 | out_error: |
| 1697 | printk(KERN_WARNING "Error: state manager failed on NFSv4 server %s" | 1714 | printk(KERN_WARNING "Error: state manager failed on NFSv4 server %s" |
diff --git a/fs/nfs/nfs4xdr.c b/fs/nfs/nfs4xdr.c index dddfb5795d7b..c3ccd2c46834 100644 --- a/fs/nfs/nfs4xdr.c +++ b/fs/nfs/nfs4xdr.c | |||
| @@ -1452,26 +1452,25 @@ static void encode_read(struct xdr_stream *xdr, const struct nfs_readargs *args, | |||
| 1452 | 1452 | ||
| 1453 | static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg *readdir, struct rpc_rqst *req, struct compound_hdr *hdr) | 1453 | static void encode_readdir(struct xdr_stream *xdr, const struct nfs4_readdir_arg *readdir, struct rpc_rqst *req, struct compound_hdr *hdr) |
| 1454 | { | 1454 | { |
| 1455 | uint32_t attrs[2] = {0, 0}; | 1455 | uint32_t attrs[2] = { |
| 1456 | FATTR4_WORD0_RDATTR_ERROR, | ||
| 1457 | FATTR4_WORD1_MOUNTED_ON_FILEID, | ||
| 1458 | }; | ||
| 1456 | uint32_t dircount = readdir->count >> 1; | 1459 | uint32_t dircount = readdir->count >> 1; |
| 1457 | __be32 *p; | 1460 | __be32 *p; |
| 1458 | 1461 | ||
| 1459 | if (readdir->plus) { | 1462 | if (readdir->plus) { |
| 1460 | attrs[0] |= FATTR4_WORD0_TYPE|FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE| | 1463 | attrs[0] |= FATTR4_WORD0_TYPE|FATTR4_WORD0_CHANGE|FATTR4_WORD0_SIZE| |
| 1461 | FATTR4_WORD0_FSID|FATTR4_WORD0_FILEHANDLE; | 1464 | FATTR4_WORD0_FSID|FATTR4_WORD0_FILEHANDLE|FATTR4_WORD0_FILEID; |
| 1462 | attrs[1] |= FATTR4_WORD1_MODE|FATTR4_WORD1_NUMLINKS|FATTR4_WORD1_OWNER| | 1465 | attrs[1] |= FATTR4_WORD1_MODE|FATTR4_WORD1_NUMLINKS|FATTR4_WORD1_OWNER| |
| 1463 | FATTR4_WORD1_OWNER_GROUP|FATTR4_WORD1_RAWDEV| | 1466 | FATTR4_WORD1_OWNER_GROUP|FATTR4_WORD1_RAWDEV| |
| 1464 | FATTR4_WORD1_SPACE_USED|FATTR4_WORD1_TIME_ACCESS| | 1467 | FATTR4_WORD1_SPACE_USED|FATTR4_WORD1_TIME_ACCESS| |
| 1465 | FATTR4_WORD1_TIME_METADATA|FATTR4_WORD1_TIME_MODIFY; | 1468 | FATTR4_WORD1_TIME_METADATA|FATTR4_WORD1_TIME_MODIFY; |
| 1466 | dircount >>= 1; | 1469 | dircount >>= 1; |
| 1467 | } | 1470 | } |
| 1468 | attrs[0] |= FATTR4_WORD0_RDATTR_ERROR|FATTR4_WORD0_FILEID; | 1471 | /* Use mounted_on_fileid only if the server supports it */ |
| 1469 | attrs[1] |= FATTR4_WORD1_MOUNTED_ON_FILEID; | 1472 | if (!(readdir->bitmask[1] & FATTR4_WORD1_MOUNTED_ON_FILEID)) |
| 1470 | /* Switch to mounted_on_fileid if the server supports it */ | 1473 | attrs[0] |= FATTR4_WORD0_FILEID; |
| 1471 | if (readdir->bitmask[1] & FATTR4_WORD1_MOUNTED_ON_FILEID) | ||
| 1472 | attrs[0] &= ~FATTR4_WORD0_FILEID; | ||
| 1473 | else | ||
| 1474 | attrs[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; | ||
| 1475 | 1474 | ||
| 1476 | p = reserve_space(xdr, 12+NFS4_VERIFIER_SIZE+20); | 1475 | p = reserve_space(xdr, 12+NFS4_VERIFIER_SIZE+20); |
| 1477 | *p++ = cpu_to_be32(OP_READDIR); | 1476 | *p++ = cpu_to_be32(OP_READDIR); |
| @@ -3140,7 +3139,7 @@ static int decode_attr_mounted_on_fileid(struct xdr_stream *xdr, uint32_t *bitma | |||
| 3140 | goto out_overflow; | 3139 | goto out_overflow; |
| 3141 | xdr_decode_hyper(p, fileid); | 3140 | xdr_decode_hyper(p, fileid); |
| 3142 | bitmap[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; | 3141 | bitmap[1] &= ~FATTR4_WORD1_MOUNTED_ON_FILEID; |
| 3143 | ret = NFS_ATTR_FATTR_FILEID; | 3142 | ret = NFS_ATTR_FATTR_MOUNTED_ON_FILEID; |
| 3144 | } | 3143 | } |
| 3145 | dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid); | 3144 | dprintk("%s: fileid=%Lu\n", __func__, (unsigned long long)*fileid); |
| 3146 | return ret; | 3145 | return ret; |
| @@ -4002,7 +4001,6 @@ static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap, | |||
| 4002 | { | 4001 | { |
| 4003 | int status; | 4002 | int status; |
| 4004 | umode_t fmode = 0; | 4003 | umode_t fmode = 0; |
| 4005 | uint64_t fileid; | ||
| 4006 | uint32_t type; | 4004 | uint32_t type; |
| 4007 | 4005 | ||
| 4008 | status = decode_attr_type(xdr, bitmap, &type); | 4006 | status = decode_attr_type(xdr, bitmap, &type); |
| @@ -4101,13 +4099,10 @@ static int decode_getfattr_attrs(struct xdr_stream *xdr, uint32_t *bitmap, | |||
| 4101 | goto xdr_error; | 4099 | goto xdr_error; |
| 4102 | fattr->valid |= status; | 4100 | fattr->valid |= status; |
| 4103 | 4101 | ||
| 4104 | status = decode_attr_mounted_on_fileid(xdr, bitmap, &fileid); | 4102 | status = decode_attr_mounted_on_fileid(xdr, bitmap, &fattr->mounted_on_fileid); |
| 4105 | if (status < 0) | 4103 | if (status < 0) |
| 4106 | goto xdr_error; | 4104 | goto xdr_error; |
| 4107 | if (status != 0 && !(fattr->valid & status)) { | 4105 | fattr->valid |= status; |
| 4108 | fattr->fileid = fileid; | ||
| 4109 | fattr->valid |= status; | ||
| 4110 | } | ||
| 4111 | 4106 | ||
| 4112 | xdr_error: | 4107 | xdr_error: |
| 4113 | dprintk("%s: xdr returned %d\n", __func__, -status); | 4108 | dprintk("%s: xdr returned %d\n", __func__, -status); |
| @@ -4838,17 +4833,21 @@ static int decode_secinfo(struct xdr_stream *xdr, struct nfs4_secinfo_res *res) | |||
| 4838 | struct nfs4_secinfo_flavor *sec_flavor; | 4833 | struct nfs4_secinfo_flavor *sec_flavor; |
| 4839 | int status; | 4834 | int status; |
| 4840 | __be32 *p; | 4835 | __be32 *p; |
| 4841 | int i; | 4836 | int i, num_flavors; |
| 4842 | 4837 | ||
| 4843 | status = decode_op_hdr(xdr, OP_SECINFO); | 4838 | status = decode_op_hdr(xdr, OP_SECINFO); |
| 4839 | if (status) | ||
| 4840 | goto out; | ||
| 4844 | p = xdr_inline_decode(xdr, 4); | 4841 | p = xdr_inline_decode(xdr, 4); |
| 4845 | if (unlikely(!p)) | 4842 | if (unlikely(!p)) |
| 4846 | goto out_overflow; | 4843 | goto out_overflow; |
| 4847 | res->flavors->num_flavors = be32_to_cpup(p); | ||
| 4848 | 4844 | ||
| 4849 | for (i = 0; i < res->flavors->num_flavors; i++) { | 4845 | res->flavors->num_flavors = 0; |
| 4846 | num_flavors = be32_to_cpup(p); | ||
| 4847 | |||
| 4848 | for (i = 0; i < num_flavors; i++) { | ||
| 4850 | sec_flavor = &res->flavors->flavors[i]; | 4849 | sec_flavor = &res->flavors->flavors[i]; |
| 4851 | if ((char *)&sec_flavor[1] - (char *)res > PAGE_SIZE) | 4850 | if ((char *)&sec_flavor[1] - (char *)res->flavors > PAGE_SIZE) |
| 4852 | break; | 4851 | break; |
| 4853 | 4852 | ||
| 4854 | p = xdr_inline_decode(xdr, 4); | 4853 | p = xdr_inline_decode(xdr, 4); |
| @@ -4857,13 +4856,15 @@ static int decode_secinfo(struct xdr_stream *xdr, struct nfs4_secinfo_res *res) | |||
| 4857 | sec_flavor->flavor = be32_to_cpup(p); | 4856 | sec_flavor->flavor = be32_to_cpup(p); |
| 4858 | 4857 | ||
| 4859 | if (sec_flavor->flavor == RPC_AUTH_GSS) { | 4858 | if (sec_flavor->flavor == RPC_AUTH_GSS) { |
| 4860 | if (decode_secinfo_gss(xdr, sec_flavor)) | 4859 | status = decode_secinfo_gss(xdr, sec_flavor); |
| 4861 | break; | 4860 | if (status) |
| 4861 | goto out; | ||
| 4862 | } | 4862 | } |
| 4863 | res->flavors->num_flavors++; | ||
| 4863 | } | 4864 | } |
| 4864 | 4865 | ||
| 4865 | return 0; | 4866 | out: |
| 4866 | 4867 | return status; | |
| 4867 | out_overflow: | 4868 | out_overflow: |
| 4868 | print_overflow_msg(__func__, xdr); | 4869 | print_overflow_msg(__func__, xdr); |
| 4869 | return -EIO; | 4870 | return -EIO; |
| @@ -6408,7 +6409,9 @@ int nfs4_decode_dirent(struct xdr_stream *xdr, struct nfs_entry *entry, | |||
| 6408 | if (decode_getfattr_attrs(xdr, bitmap, entry->fattr, entry->fh, | 6409 | if (decode_getfattr_attrs(xdr, bitmap, entry->fattr, entry->fh, |
| 6409 | entry->server, 1) < 0) | 6410 | entry->server, 1) < 0) |
| 6410 | goto out_overflow; | 6411 | goto out_overflow; |
| 6411 | if (entry->fattr->valid & NFS_ATTR_FATTR_FILEID) | 6412 | if (entry->fattr->valid & NFS_ATTR_FATTR_MOUNTED_ON_FILEID) |
| 6413 | entry->ino = entry->fattr->mounted_on_fileid; | ||
| 6414 | else if (entry->fattr->valid & NFS_ATTR_FATTR_FILEID) | ||
| 6412 | entry->ino = entry->fattr->fileid; | 6415 | entry->ino = entry->fattr->fileid; |
| 6413 | 6416 | ||
| 6414 | entry->d_type = DT_UNKNOWN; | 6417 | entry->d_type = DT_UNKNOWN; |
diff --git a/fs/nfs/pnfs.c b/fs/nfs/pnfs.c index d9ab97269ce6..ff681ab65d31 100644 --- a/fs/nfs/pnfs.c +++ b/fs/nfs/pnfs.c | |||
| @@ -1004,6 +1004,7 @@ pnfs_set_layoutcommit(struct nfs_write_data *wdata) | |||
| 1004 | { | 1004 | { |
| 1005 | struct nfs_inode *nfsi = NFS_I(wdata->inode); | 1005 | struct nfs_inode *nfsi = NFS_I(wdata->inode); |
| 1006 | loff_t end_pos = wdata->args.offset + wdata->res.count; | 1006 | loff_t end_pos = wdata->args.offset + wdata->res.count; |
| 1007 | bool mark_as_dirty = false; | ||
| 1007 | 1008 | ||
| 1008 | spin_lock(&nfsi->vfs_inode.i_lock); | 1009 | spin_lock(&nfsi->vfs_inode.i_lock); |
| 1009 | if (!test_and_set_bit(NFS_INO_LAYOUTCOMMIT, &nfsi->flags)) { | 1010 | if (!test_and_set_bit(NFS_INO_LAYOUTCOMMIT, &nfsi->flags)) { |
| @@ -1011,13 +1012,18 @@ pnfs_set_layoutcommit(struct nfs_write_data *wdata) | |||
| 1011 | get_lseg(wdata->lseg); | 1012 | get_lseg(wdata->lseg); |
| 1012 | wdata->lseg->pls_lc_cred = | 1013 | wdata->lseg->pls_lc_cred = |
| 1013 | get_rpccred(wdata->args.context->state->owner->so_cred); | 1014 | get_rpccred(wdata->args.context->state->owner->so_cred); |
| 1014 | mark_inode_dirty_sync(wdata->inode); | 1015 | mark_as_dirty = true; |
| 1015 | dprintk("%s: Set layoutcommit for inode %lu ", | 1016 | dprintk("%s: Set layoutcommit for inode %lu ", |
| 1016 | __func__, wdata->inode->i_ino); | 1017 | __func__, wdata->inode->i_ino); |
| 1017 | } | 1018 | } |
| 1018 | if (end_pos > wdata->lseg->pls_end_pos) | 1019 | if (end_pos > wdata->lseg->pls_end_pos) |
| 1019 | wdata->lseg->pls_end_pos = end_pos; | 1020 | wdata->lseg->pls_end_pos = end_pos; |
| 1020 | spin_unlock(&nfsi->vfs_inode.i_lock); | 1021 | spin_unlock(&nfsi->vfs_inode.i_lock); |
| 1022 | |||
| 1023 | /* if pnfs_layoutcommit_inode() runs between inode locks, the next one | ||
| 1024 | * will be a noop because NFS_INO_LAYOUTCOMMIT will not be set */ | ||
| 1025 | if (mark_as_dirty) | ||
| 1026 | mark_inode_dirty_sync(wdata->inode); | ||
| 1021 | } | 1027 | } |
| 1022 | EXPORT_SYMBOL_GPL(pnfs_set_layoutcommit); | 1028 | EXPORT_SYMBOL_GPL(pnfs_set_layoutcommit); |
| 1023 | 1029 | ||
diff --git a/fs/nfs/super.c b/fs/nfs/super.c index 2b8e9a5e366a..e288f06d3fa7 100644 --- a/fs/nfs/super.c +++ b/fs/nfs/super.c | |||
| @@ -1004,6 +1004,7 @@ static int nfs_parse_security_flavors(char *value, | |||
| 1004 | return 0; | 1004 | return 0; |
| 1005 | } | 1005 | } |
| 1006 | 1006 | ||
| 1007 | mnt->flags |= NFS_MOUNT_SECFLAVOUR; | ||
| 1007 | mnt->auth_flavor_len = 1; | 1008 | mnt->auth_flavor_len = 1; |
| 1008 | return 1; | 1009 | return 1; |
| 1009 | } | 1010 | } |
| @@ -1976,6 +1977,15 @@ nfs_remount(struct super_block *sb, int *flags, char *raw_data) | |||
| 1976 | if (error < 0) | 1977 | if (error < 0) |
| 1977 | goto out; | 1978 | goto out; |
| 1978 | 1979 | ||
| 1980 | /* | ||
| 1981 | * noac is a special case. It implies -o sync, but that's not | ||
| 1982 | * necessarily reflected in the mtab options. do_remount_sb | ||
| 1983 | * will clear MS_SYNCHRONOUS if -o sync wasn't specified in the | ||
| 1984 | * remount options, so we have to explicitly reset it. | ||
| 1985 | */ | ||
| 1986 | if (data->flags & NFS_MOUNT_NOAC) | ||
| 1987 | *flags |= MS_SYNCHRONOUS; | ||
| 1988 | |||
| 1979 | /* compare new mount options with old ones */ | 1989 | /* compare new mount options with old ones */ |
| 1980 | error = nfs_compare_remount_data(nfss, data); | 1990 | error = nfs_compare_remount_data(nfss, data); |
| 1981 | out: | 1991 | out: |
| @@ -2235,8 +2245,7 @@ static struct dentry *nfs_fs_mount(struct file_system_type *fs_type, | |||
| 2235 | if (!s->s_root) { | 2245 | if (!s->s_root) { |
| 2236 | /* initial superblock/root creation */ | 2246 | /* initial superblock/root creation */ |
| 2237 | nfs_fill_super(s, data); | 2247 | nfs_fill_super(s, data); |
| 2238 | nfs_fscache_get_super_cookie( | 2248 | nfs_fscache_get_super_cookie(s, data->fscache_uniq, NULL); |
| 2239 | s, data ? data->fscache_uniq : NULL, NULL); | ||
| 2240 | } | 2249 | } |
| 2241 | 2250 | ||
| 2242 | mntroot = nfs_get_root(s, mntfh, dev_name); | 2251 | mntroot = nfs_get_root(s, mntfh, dev_name); |
diff --git a/fs/nfs/write.c b/fs/nfs/write.c index e4cbc11a74ab..3bd5d7e80f6c 100644 --- a/fs/nfs/write.c +++ b/fs/nfs/write.c | |||
| @@ -680,7 +680,6 @@ static int nfs_writepage_setup(struct nfs_open_context *ctx, struct page *page, | |||
| 680 | req = nfs_setup_write_request(ctx, page, offset, count); | 680 | req = nfs_setup_write_request(ctx, page, offset, count); |
| 681 | if (IS_ERR(req)) | 681 | if (IS_ERR(req)) |
| 682 | return PTR_ERR(req); | 682 | return PTR_ERR(req); |
| 683 | nfs_mark_request_dirty(req); | ||
| 684 | /* Update file length */ | 683 | /* Update file length */ |
| 685 | nfs_grow_file(page, offset, count); | 684 | nfs_grow_file(page, offset, count); |
| 686 | nfs_mark_uptodate(page, req->wb_pgbase, req->wb_bytes); | 685 | nfs_mark_uptodate(page, req->wb_pgbase, req->wb_bytes); |
| @@ -1418,8 +1417,7 @@ static void nfs_commit_done(struct rpc_task *task, void *calldata) | |||
| 1418 | task->tk_pid, task->tk_status); | 1417 | task->tk_pid, task->tk_status); |
| 1419 | 1418 | ||
| 1420 | /* Call the NFS version-specific code */ | 1419 | /* Call the NFS version-specific code */ |
| 1421 | if (NFS_PROTO(data->inode)->commit_done(task, data) != 0) | 1420 | NFS_PROTO(data->inode)->commit_done(task, data); |
| 1422 | return; | ||
| 1423 | } | 1421 | } |
| 1424 | 1422 | ||
| 1425 | void nfs_commit_release_pages(struct nfs_write_data *data) | 1423 | void nfs_commit_release_pages(struct nfs_write_data *data) |
diff --git a/fs/ocfs2/ocfs2_fs.h b/fs/ocfs2/ocfs2_fs.h index b68f87a83924..938387a10d5d 100644 --- a/fs/ocfs2/ocfs2_fs.h +++ b/fs/ocfs2/ocfs2_fs.h | |||
| @@ -1019,7 +1019,7 @@ struct ocfs2_xattr_entry { | |||
| 1019 | __le16 xe_name_offset; /* byte offset from the 1st entry in the | 1019 | __le16 xe_name_offset; /* byte offset from the 1st entry in the |
| 1020 | local xattr storage(inode, xattr block or | 1020 | local xattr storage(inode, xattr block or |
| 1021 | xattr bucket). */ | 1021 | xattr bucket). */ |
| 1022 | __u8 xe_name_len; /* xattr name len, does't include prefix. */ | 1022 | __u8 xe_name_len; /* xattr name len, doesn't include prefix. */ |
| 1023 | __u8 xe_type; /* the low 7 bits indicate the name prefix | 1023 | __u8 xe_type; /* the low 7 bits indicate the name prefix |
| 1024 | * type and the highest bit indicates whether | 1024 | * type and the highest bit indicates whether |
| 1025 | * the EA is stored in the local storage. */ | 1025 | * the EA is stored in the local storage. */ |
diff --git a/fs/ubifs/log.c b/fs/ubifs/log.c index 4d0cb1241460..40fa780ebea7 100644 --- a/fs/ubifs/log.c +++ b/fs/ubifs/log.c | |||
| @@ -175,26 +175,6 @@ void ubifs_add_bud(struct ubifs_info *c, struct ubifs_bud *bud) | |||
| 175 | } | 175 | } |
| 176 | 176 | ||
| 177 | /** | 177 | /** |
| 178 | * ubifs_create_buds_lists - create journal head buds lists for remount rw. | ||
| 179 | * @c: UBIFS file-system description object | ||
| 180 | */ | ||
| 181 | void ubifs_create_buds_lists(struct ubifs_info *c) | ||
| 182 | { | ||
| 183 | struct rb_node *p; | ||
| 184 | |||
| 185 | spin_lock(&c->buds_lock); | ||
| 186 | p = rb_first(&c->buds); | ||
| 187 | while (p) { | ||
| 188 | struct ubifs_bud *bud = rb_entry(p, struct ubifs_bud, rb); | ||
| 189 | struct ubifs_jhead *jhead = &c->jheads[bud->jhead]; | ||
| 190 | |||
| 191 | list_add_tail(&bud->list, &jhead->buds_list); | ||
| 192 | p = rb_next(p); | ||
| 193 | } | ||
| 194 | spin_unlock(&c->buds_lock); | ||
| 195 | } | ||
| 196 | |||
| 197 | /** | ||
| 198 | * ubifs_add_bud_to_log - add a new bud to the log. | 178 | * ubifs_add_bud_to_log - add a new bud to the log. |
| 199 | * @c: UBIFS file-system description object | 179 | * @c: UBIFS file-system description object |
| 200 | * @jhead: journal head the bud belongs to | 180 | * @jhead: journal head the bud belongs to |
diff --git a/fs/ubifs/replay.c b/fs/ubifs/replay.c index eed0fcff8d73..d3d6d365bfc1 100644 --- a/fs/ubifs/replay.c +++ b/fs/ubifs/replay.c | |||
| @@ -59,6 +59,7 @@ enum { | |||
| 59 | * @new_size: truncation new size | 59 | * @new_size: truncation new size |
| 60 | * @free: amount of free space in a bud | 60 | * @free: amount of free space in a bud |
| 61 | * @dirty: amount of dirty space in a bud from padding and deletion nodes | 61 | * @dirty: amount of dirty space in a bud from padding and deletion nodes |
| 62 | * @jhead: journal head number of the bud | ||
| 62 | * | 63 | * |
| 63 | * UBIFS journal replay must compare node sequence numbers, which means it must | 64 | * UBIFS journal replay must compare node sequence numbers, which means it must |
| 64 | * build a tree of node information to insert into the TNC. | 65 | * build a tree of node information to insert into the TNC. |
| @@ -80,6 +81,7 @@ struct replay_entry { | |||
| 80 | struct { | 81 | struct { |
| 81 | int free; | 82 | int free; |
| 82 | int dirty; | 83 | int dirty; |
| 84 | int jhead; | ||
| 83 | }; | 85 | }; |
| 84 | }; | 86 | }; |
| 85 | }; | 87 | }; |
| @@ -159,6 +161,11 @@ static int set_bud_lprops(struct ubifs_info *c, struct replay_entry *r) | |||
| 159 | err = PTR_ERR(lp); | 161 | err = PTR_ERR(lp); |
| 160 | goto out; | 162 | goto out; |
| 161 | } | 163 | } |
| 164 | |||
| 165 | /* Make sure the journal head points to the latest bud */ | ||
| 166 | err = ubifs_wbuf_seek_nolock(&c->jheads[r->jhead].wbuf, r->lnum, | ||
| 167 | c->leb_size - r->free, UBI_SHORTTERM); | ||
| 168 | |||
| 162 | out: | 169 | out: |
| 163 | ubifs_release_lprops(c); | 170 | ubifs_release_lprops(c); |
| 164 | return err; | 171 | return err; |
| @@ -627,10 +634,6 @@ static int replay_bud(struct ubifs_info *c, int lnum, int offs, int jhead, | |||
| 627 | ubifs_assert(sleb->endpt - offs >= used); | 634 | ubifs_assert(sleb->endpt - offs >= used); |
| 628 | ubifs_assert(sleb->endpt % c->min_io_size == 0); | 635 | ubifs_assert(sleb->endpt % c->min_io_size == 0); |
| 629 | 636 | ||
| 630 | if (sleb->endpt + c->min_io_size <= c->leb_size && !c->ro_mount) | ||
| 631 | err = ubifs_wbuf_seek_nolock(&c->jheads[jhead].wbuf, lnum, | ||
| 632 | sleb->endpt, UBI_SHORTTERM); | ||
| 633 | |||
| 634 | *dirty = sleb->endpt - offs - used; | 637 | *dirty = sleb->endpt - offs - used; |
| 635 | *free = c->leb_size - sleb->endpt; | 638 | *free = c->leb_size - sleb->endpt; |
| 636 | 639 | ||
| @@ -653,12 +656,14 @@ out_dump: | |||
| 653 | * @sqnum: sequence number | 656 | * @sqnum: sequence number |
| 654 | * @free: amount of free space in bud | 657 | * @free: amount of free space in bud |
| 655 | * @dirty: amount of dirty space from padding and deletion nodes | 658 | * @dirty: amount of dirty space from padding and deletion nodes |
| 659 | * @jhead: journal head number for the bud | ||
| 656 | * | 660 | * |
| 657 | * This function inserts a reference node to the replay tree and returns zero | 661 | * This function inserts a reference node to the replay tree and returns zero |
| 658 | * in case of success or a negative error code in case of failure. | 662 | * in case of success or a negative error code in case of failure. |
| 659 | */ | 663 | */ |
| 660 | static int insert_ref_node(struct ubifs_info *c, int lnum, int offs, | 664 | static int insert_ref_node(struct ubifs_info *c, int lnum, int offs, |
| 661 | unsigned long long sqnum, int free, int dirty) | 665 | unsigned long long sqnum, int free, int dirty, |
| 666 | int jhead) | ||
| 662 | { | 667 | { |
| 663 | struct rb_node **p = &c->replay_tree.rb_node, *parent = NULL; | 668 | struct rb_node **p = &c->replay_tree.rb_node, *parent = NULL; |
| 664 | struct replay_entry *r; | 669 | struct replay_entry *r; |
| @@ -688,6 +693,7 @@ static int insert_ref_node(struct ubifs_info *c, int lnum, int offs, | |||
| 688 | r->flags = REPLAY_REF; | 693 | r->flags = REPLAY_REF; |
| 689 | r->free = free; | 694 | r->free = free; |
| 690 | r->dirty = dirty; | 695 | r->dirty = dirty; |
| 696 | r->jhead = jhead; | ||
| 691 | 697 | ||
| 692 | rb_link_node(&r->rb, parent, p); | 698 | rb_link_node(&r->rb, parent, p); |
| 693 | rb_insert_color(&r->rb, &c->replay_tree); | 699 | rb_insert_color(&r->rb, &c->replay_tree); |
| @@ -712,7 +718,7 @@ static int replay_buds(struct ubifs_info *c) | |||
| 712 | if (err) | 718 | if (err) |
| 713 | return err; | 719 | return err; |
| 714 | err = insert_ref_node(c, b->bud->lnum, b->bud->start, b->sqnum, | 720 | err = insert_ref_node(c, b->bud->lnum, b->bud->start, b->sqnum, |
| 715 | free, dirty); | 721 | free, dirty, b->bud->jhead); |
| 716 | if (err) | 722 | if (err) |
| 717 | return err; | 723 | return err; |
| 718 | } | 724 | } |
diff --git a/fs/ubifs/super.c b/fs/ubifs/super.c index be6c7b008f38..04ad07f4fcc3 100644 --- a/fs/ubifs/super.c +++ b/fs/ubifs/super.c | |||
| @@ -1257,12 +1257,12 @@ static int mount_ubifs(struct ubifs_info *c) | |||
| 1257 | goto out_free; | 1257 | goto out_free; |
| 1258 | } | 1258 | } |
| 1259 | 1259 | ||
| 1260 | err = alloc_wbufs(c); | ||
| 1261 | if (err) | ||
| 1262 | goto out_cbuf; | ||
| 1263 | |||
| 1260 | sprintf(c->bgt_name, BGT_NAME_PATTERN, c->vi.ubi_num, c->vi.vol_id); | 1264 | sprintf(c->bgt_name, BGT_NAME_PATTERN, c->vi.ubi_num, c->vi.vol_id); |
| 1261 | if (!c->ro_mount) { | 1265 | if (!c->ro_mount) { |
| 1262 | err = alloc_wbufs(c); | ||
| 1263 | if (err) | ||
| 1264 | goto out_cbuf; | ||
| 1265 | |||
| 1266 | /* Create background thread */ | 1266 | /* Create background thread */ |
| 1267 | c->bgt = kthread_create(ubifs_bg_thread, c, "%s", c->bgt_name); | 1267 | c->bgt = kthread_create(ubifs_bg_thread, c, "%s", c->bgt_name); |
| 1268 | if (IS_ERR(c->bgt)) { | 1268 | if (IS_ERR(c->bgt)) { |
| @@ -1631,12 +1631,6 @@ static int ubifs_remount_rw(struct ubifs_info *c) | |||
| 1631 | if (err) | 1631 | if (err) |
| 1632 | goto out; | 1632 | goto out; |
| 1633 | 1633 | ||
| 1634 | err = alloc_wbufs(c); | ||
| 1635 | if (err) | ||
| 1636 | goto out; | ||
| 1637 | |||
| 1638 | ubifs_create_buds_lists(c); | ||
| 1639 | |||
| 1640 | /* Create background thread */ | 1634 | /* Create background thread */ |
| 1641 | c->bgt = kthread_create(ubifs_bg_thread, c, "%s", c->bgt_name); | 1635 | c->bgt = kthread_create(ubifs_bg_thread, c, "%s", c->bgt_name); |
| 1642 | if (IS_ERR(c->bgt)) { | 1636 | if (IS_ERR(c->bgt)) { |
| @@ -1744,7 +1738,6 @@ static void ubifs_remount_ro(struct ubifs_info *c) | |||
| 1744 | if (err) | 1738 | if (err) |
| 1745 | ubifs_ro_mode(c, err); | 1739 | ubifs_ro_mode(c, err); |
| 1746 | 1740 | ||
| 1747 | free_wbufs(c); | ||
| 1748 | vfree(c->orph_buf); | 1741 | vfree(c->orph_buf); |
| 1749 | c->orph_buf = NULL; | 1742 | c->orph_buf = NULL; |
| 1750 | kfree(c->write_reserve_buf); | 1743 | kfree(c->write_reserve_buf); |
diff --git a/include/drm/drm_pciids.h b/include/drm/drm_pciids.h index 816e30cbd968..f04b2a3b0f49 100644 --- a/include/drm/drm_pciids.h +++ b/include/drm/drm_pciids.h | |||
| @@ -155,6 +155,7 @@ | |||
| 155 | {0x1002, 0x6719, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ | 155 | {0x1002, 0x6719, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ |
| 156 | {0x1002, 0x671c, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ | 156 | {0x1002, 0x671c, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ |
| 157 | {0x1002, 0x671d, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ | 157 | {0x1002, 0x671d, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ |
| 158 | {0x1002, 0x671f, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CAYMAN|RADEON_NEW_MEMMAP}, \ | ||
| 158 | {0x1002, 0x6720, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 159 | {0x1002, 0x6720, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 159 | {0x1002, 0x6721, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 160 | {0x1002, 0x6721, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 160 | {0x1002, 0x6722, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ | 161 | {0x1002, 0x6722, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ |
| @@ -167,6 +168,7 @@ | |||
| 167 | {0x1002, 0x6729, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ | 168 | {0x1002, 0x6729, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ |
| 168 | {0x1002, 0x6738, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ | 169 | {0x1002, 0x6738, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ |
| 169 | {0x1002, 0x6739, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ | 170 | {0x1002, 0x6739, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ |
| 171 | {0x1002, 0x673e, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_BARTS|RADEON_NEW_MEMMAP}, \ | ||
| 170 | {0x1002, 0x6740, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_TURKS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 172 | {0x1002, 0x6740, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_TURKS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 171 | {0x1002, 0x6741, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_TURKS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 173 | {0x1002, 0x6741, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_TURKS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 172 | {0x1002, 0x6742, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_TURKS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 174 | {0x1002, 0x6742, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_TURKS|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| @@ -199,6 +201,7 @@ | |||
| 199 | {0x1002, 0x688D, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ | 201 | {0x1002, 0x688D, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ |
| 200 | {0x1002, 0x6898, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ | 202 | {0x1002, 0x6898, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ |
| 201 | {0x1002, 0x6899, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ | 203 | {0x1002, 0x6899, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ |
| 204 | {0x1002, 0x689b, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ | ||
| 202 | {0x1002, 0x689c, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_HEMLOCK|RADEON_NEW_MEMMAP}, \ | 205 | {0x1002, 0x689c, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_HEMLOCK|RADEON_NEW_MEMMAP}, \ |
| 203 | {0x1002, 0x689d, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_HEMLOCK|RADEON_NEW_MEMMAP}, \ | 206 | {0x1002, 0x689d, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_HEMLOCK|RADEON_NEW_MEMMAP}, \ |
| 204 | {0x1002, 0x689e, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ | 207 | {0x1002, 0x689e, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_CYPRESS|RADEON_NEW_MEMMAP}, \ |
| @@ -209,7 +212,9 @@ | |||
| 209 | {0x1002, 0x68b0, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 212 | {0x1002, 0x68b0, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 210 | {0x1002, 0x68b8, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ | 213 | {0x1002, 0x68b8, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ |
| 211 | {0x1002, 0x68b9, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ | 214 | {0x1002, 0x68b9, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ |
| 215 | {0x1002, 0x68ba, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ | ||
| 212 | {0x1002, 0x68be, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ | 216 | {0x1002, 0x68be, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ |
| 217 | {0x1002, 0x68bf, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_JUNIPER|RADEON_NEW_MEMMAP}, \ | ||
| 213 | {0x1002, 0x68c0, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_REDWOOD|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 218 | {0x1002, 0x68c0, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_REDWOOD|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 214 | {0x1002, 0x68c1, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_REDWOOD|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 219 | {0x1002, 0x68c1, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_REDWOOD|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
| 215 | {0x1002, 0x68c7, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_REDWOOD|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ | 220 | {0x1002, 0x68c7, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_REDWOOD|RADEON_IS_MOBILITY|RADEON_NEW_MEMMAP}, \ |
diff --git a/include/drm/radeon_drm.h b/include/drm/radeon_drm.h index 7aa5dddb2098..787f7b6fd622 100644 --- a/include/drm/radeon_drm.h +++ b/include/drm/radeon_drm.h | |||
| @@ -910,6 +910,7 @@ struct drm_radeon_cs { | |||
| 910 | #define RADEON_INFO_CLOCK_CRYSTAL_FREQ 0x09 /* clock crystal frequency */ | 910 | #define RADEON_INFO_CLOCK_CRYSTAL_FREQ 0x09 /* clock crystal frequency */ |
| 911 | #define RADEON_INFO_NUM_BACKENDS 0x0a /* DB/backends for r600+ - need for OQ */ | 911 | #define RADEON_INFO_NUM_BACKENDS 0x0a /* DB/backends for r600+ - need for OQ */ |
| 912 | #define RADEON_INFO_NUM_TILE_PIPES 0x0b /* tile pipes for r600+ */ | 912 | #define RADEON_INFO_NUM_TILE_PIPES 0x0b /* tile pipes for r600+ */ |
| 913 | #define RADEON_INFO_FUSION_GART_WORKING 0x0c /* fusion writes to GTT were broken before this */ | ||
| 913 | 914 | ||
| 914 | struct drm_radeon_info { | 915 | struct drm_radeon_info { |
| 915 | uint32_t request; | 916 | uint32_t request; |
diff --git a/include/linux/huge_mm.h b/include/linux/huge_mm.h index df29c8fde36b..8847c8c29791 100644 --- a/include/linux/huge_mm.h +++ b/include/linux/huge_mm.h | |||
| @@ -117,7 +117,7 @@ static inline void vma_adjust_trans_huge(struct vm_area_struct *vma, | |||
| 117 | unsigned long end, | 117 | unsigned long end, |
| 118 | long adjust_next) | 118 | long adjust_next) |
| 119 | { | 119 | { |
| 120 | if (!vma->anon_vma || vma->vm_ops || vma->vm_file) | 120 | if (!vma->anon_vma || vma->vm_ops) |
| 121 | return; | 121 | return; |
| 122 | __vma_adjust_trans_huge(vma, start, end, adjust_next); | 122 | __vma_adjust_trans_huge(vma, start, end, adjust_next); |
| 123 | } | 123 | } |
diff --git a/include/linux/mfd/wm831x/pdata.h b/include/linux/mfd/wm831x/pdata.h index afe4db49402d..632d1567a1b6 100644 --- a/include/linux/mfd/wm831x/pdata.h +++ b/include/linux/mfd/wm831x/pdata.h | |||
| @@ -81,7 +81,9 @@ struct wm831x_touch_pdata { | |||
| 81 | int rpu; /** Pen down sensitivity resistor divider */ | 81 | int rpu; /** Pen down sensitivity resistor divider */ |
| 82 | int pressure; /** Report pressure (boolean) */ | 82 | int pressure; /** Report pressure (boolean) */ |
| 83 | unsigned int data_irq; /** Touch data ready IRQ */ | 83 | unsigned int data_irq; /** Touch data ready IRQ */ |
| 84 | int data_irqf; /** IRQ flags for data ready IRQ */ | ||
| 84 | unsigned int pd_irq; /** Touch pendown detect IRQ */ | 85 | unsigned int pd_irq; /** Touch pendown detect IRQ */ |
| 86 | int pd_irqf; /** IRQ flags for pen down IRQ */ | ||
| 85 | }; | 87 | }; |
| 86 | 88 | ||
| 87 | enum wm831x_watchdog_action { | 89 | enum wm831x_watchdog_action { |
diff --git a/include/linux/mm.h b/include/linux/mm.h index 692dbae6ffa7..2348db26bc3d 100644 --- a/include/linux/mm.h +++ b/include/linux/mm.h | |||
| @@ -137,7 +137,8 @@ extern unsigned int kobjsize(const void *objp); | |||
| 137 | #define VM_RandomReadHint(v) ((v)->vm_flags & VM_RAND_READ) | 137 | #define VM_RandomReadHint(v) ((v)->vm_flags & VM_RAND_READ) |
| 138 | 138 | ||
| 139 | /* | 139 | /* |
| 140 | * special vmas that are non-mergable, non-mlock()able | 140 | * Special vmas that are non-mergable, non-mlock()able. |
| 141 | * Note: mm/huge_memory.c VM_NO_THP depends on this definition. | ||
| 141 | */ | 142 | */ |
| 142 | #define VM_SPECIAL (VM_IO | VM_DONTEXPAND | VM_RESERVED | VM_PFNMAP) | 143 | #define VM_SPECIAL (VM_IO | VM_DONTEXPAND | VM_RESERVED | VM_PFNMAP) |
| 143 | 144 | ||
diff --git a/include/linux/mmc/host.h b/include/linux/mmc/host.h index bcb793ec7374..eb792cb6d745 100644 --- a/include/linux/mmc/host.h +++ b/include/linux/mmc/host.h | |||
| @@ -183,7 +183,6 @@ struct mmc_host { | |||
| 183 | struct work_struct clk_gate_work; /* delayed clock gate */ | 183 | struct work_struct clk_gate_work; /* delayed clock gate */ |
| 184 | unsigned int clk_old; /* old clock value cache */ | 184 | unsigned int clk_old; /* old clock value cache */ |
| 185 | spinlock_t clk_lock; /* lock for clk fields */ | 185 | spinlock_t clk_lock; /* lock for clk fields */ |
| 186 | struct mutex clk_gate_mutex; /* mutex for clock gating */ | ||
| 187 | #endif | 186 | #endif |
| 188 | 187 | ||
| 189 | /* host specific block data */ | 188 | /* host specific block data */ |
diff --git a/include/linux/nfs_fs_sb.h b/include/linux/nfs_fs_sb.h index 216cea5db0aa..87694ca86914 100644 --- a/include/linux/nfs_fs_sb.h +++ b/include/linux/nfs_fs_sb.h | |||
| @@ -47,6 +47,7 @@ struct nfs_client { | |||
| 47 | 47 | ||
| 48 | #ifdef CONFIG_NFS_V4 | 48 | #ifdef CONFIG_NFS_V4 |
| 49 | u64 cl_clientid; /* constant */ | 49 | u64 cl_clientid; /* constant */ |
| 50 | nfs4_verifier cl_confirm; /* Clientid verifier */ | ||
| 50 | unsigned long cl_state; | 51 | unsigned long cl_state; |
| 51 | 52 | ||
| 52 | spinlock_t cl_lock; | 53 | spinlock_t cl_lock; |
diff --git a/include/linux/nfs_xdr.h b/include/linux/nfs_xdr.h index 78b101e487ea..890dce242639 100644 --- a/include/linux/nfs_xdr.h +++ b/include/linux/nfs_xdr.h | |||
| @@ -50,6 +50,7 @@ struct nfs_fattr { | |||
| 50 | } du; | 50 | } du; |
| 51 | struct nfs_fsid fsid; | 51 | struct nfs_fsid fsid; |
| 52 | __u64 fileid; | 52 | __u64 fileid; |
| 53 | __u64 mounted_on_fileid; | ||
| 53 | struct timespec atime; | 54 | struct timespec atime; |
| 54 | struct timespec mtime; | 55 | struct timespec mtime; |
| 55 | struct timespec ctime; | 56 | struct timespec ctime; |
| @@ -83,6 +84,7 @@ struct nfs_fattr { | |||
| 83 | #define NFS_ATTR_FATTR_PRECHANGE (1U << 18) | 84 | #define NFS_ATTR_FATTR_PRECHANGE (1U << 18) |
| 84 | #define NFS_ATTR_FATTR_V4_REFERRAL (1U << 19) /* NFSv4 referral */ | 85 | #define NFS_ATTR_FATTR_V4_REFERRAL (1U << 19) /* NFSv4 referral */ |
| 85 | #define NFS_ATTR_FATTR_MOUNTPOINT (1U << 20) /* Treat as mountpoint */ | 86 | #define NFS_ATTR_FATTR_MOUNTPOINT (1U << 20) /* Treat as mountpoint */ |
| 87 | #define NFS_ATTR_FATTR_MOUNTED_ON_FILEID (1U << 21) | ||
| 86 | 88 | ||
| 87 | #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \ | 89 | #define NFS_ATTR_FATTR (NFS_ATTR_FATTR_TYPE \ |
| 88 | | NFS_ATTR_FATTR_MODE \ | 90 | | NFS_ATTR_FATTR_MODE \ |
diff --git a/include/linux/pci_ids.h b/include/linux/pci_ids.h index 4e2c9150a785..8abe8d78c4bf 100644 --- a/include/linux/pci_ids.h +++ b/include/linux/pci_ids.h | |||
| @@ -2477,15 +2477,12 @@ | |||
| 2477 | #define PCI_DEVICE_ID_INTEL_82840_HB 0x1a21 | 2477 | #define PCI_DEVICE_ID_INTEL_82840_HB 0x1a21 |
| 2478 | #define PCI_DEVICE_ID_INTEL_82845_HB 0x1a30 | 2478 | #define PCI_DEVICE_ID_INTEL_82845_HB 0x1a30 |
| 2479 | #define PCI_DEVICE_ID_INTEL_IOAT 0x1a38 | 2479 | #define PCI_DEVICE_ID_INTEL_IOAT 0x1a38 |
| 2480 | #define PCI_DEVICE_ID_INTEL_COUGARPOINT_SMBUS 0x1c22 | ||
| 2481 | #define PCI_DEVICE_ID_INTEL_COUGARPOINT_LPC_MIN 0x1c41 | 2480 | #define PCI_DEVICE_ID_INTEL_COUGARPOINT_LPC_MIN 0x1c41 |
| 2482 | #define PCI_DEVICE_ID_INTEL_COUGARPOINT_LPC_MAX 0x1c5f | 2481 | #define PCI_DEVICE_ID_INTEL_COUGARPOINT_LPC_MAX 0x1c5f |
| 2483 | #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS 0x1d22 | ||
| 2484 | #define PCI_DEVICE_ID_INTEL_PATSBURG_LPC_0 0x1d40 | 2482 | #define PCI_DEVICE_ID_INTEL_PATSBURG_LPC_0 0x1d40 |
| 2485 | #define PCI_DEVICE_ID_INTEL_PATSBURG_LPC_1 0x1d41 | 2483 | #define PCI_DEVICE_ID_INTEL_PATSBURG_LPC_1 0x1d41 |
| 2486 | #define PCI_DEVICE_ID_INTEL_DH89XXCC_LPC_MIN 0x2310 | 2484 | #define PCI_DEVICE_ID_INTEL_DH89XXCC_LPC_MIN 0x2310 |
| 2487 | #define PCI_DEVICE_ID_INTEL_DH89XXCC_LPC_MAX 0x231f | 2485 | #define PCI_DEVICE_ID_INTEL_DH89XXCC_LPC_MAX 0x231f |
| 2488 | #define PCI_DEVICE_ID_INTEL_DH89XXCC_SMBUS 0x2330 | ||
| 2489 | #define PCI_DEVICE_ID_INTEL_82801AA_0 0x2410 | 2486 | #define PCI_DEVICE_ID_INTEL_82801AA_0 0x2410 |
| 2490 | #define PCI_DEVICE_ID_INTEL_82801AA_1 0x2411 | 2487 | #define PCI_DEVICE_ID_INTEL_82801AA_1 0x2411 |
| 2491 | #define PCI_DEVICE_ID_INTEL_82801AA_3 0x2413 | 2488 | #define PCI_DEVICE_ID_INTEL_82801AA_3 0x2413 |
| @@ -2696,7 +2693,6 @@ | |||
| 2696 | #define PCI_DEVICE_ID_INTEL_ICH10_5 0x3a60 | 2693 | #define PCI_DEVICE_ID_INTEL_ICH10_5 0x3a60 |
| 2697 | #define PCI_DEVICE_ID_INTEL_5_3400_SERIES_LPC_MIN 0x3b00 | 2694 | #define PCI_DEVICE_ID_INTEL_5_3400_SERIES_LPC_MIN 0x3b00 |
| 2698 | #define PCI_DEVICE_ID_INTEL_5_3400_SERIES_LPC_MAX 0x3b1f | 2695 | #define PCI_DEVICE_ID_INTEL_5_3400_SERIES_LPC_MAX 0x3b1f |
| 2699 | #define PCI_DEVICE_ID_INTEL_5_3400_SERIES_SMBUS 0x3b30 | ||
| 2700 | #define PCI_DEVICE_ID_INTEL_IOAT_SNB 0x402f | 2696 | #define PCI_DEVICE_ID_INTEL_IOAT_SNB 0x402f |
| 2701 | #define PCI_DEVICE_ID_INTEL_5100_16 0x65f0 | 2697 | #define PCI_DEVICE_ID_INTEL_5100_16 0x65f0 |
| 2702 | #define PCI_DEVICE_ID_INTEL_5100_21 0x65f5 | 2698 | #define PCI_DEVICE_ID_INTEL_5100_21 0x65f5 |
diff --git a/include/linux/sunrpc/sched.h b/include/linux/sunrpc/sched.h index d81db8012c63..f73c482ec9c6 100644 --- a/include/linux/sunrpc/sched.h +++ b/include/linux/sunrpc/sched.h | |||
| @@ -127,13 +127,16 @@ struct rpc_task_setup { | |||
| 127 | #define RPC_TASK_KILLED 0x0100 /* task was killed */ | 127 | #define RPC_TASK_KILLED 0x0100 /* task was killed */ |
| 128 | #define RPC_TASK_SOFT 0x0200 /* Use soft timeouts */ | 128 | #define RPC_TASK_SOFT 0x0200 /* Use soft timeouts */ |
| 129 | #define RPC_TASK_SOFTCONN 0x0400 /* Fail if can't connect */ | 129 | #define RPC_TASK_SOFTCONN 0x0400 /* Fail if can't connect */ |
| 130 | #define RPC_TASK_SENT 0x0800 /* message was sent */ | ||
| 131 | #define RPC_TASK_TIMEOUT 0x1000 /* fail with ETIMEDOUT on timeout */ | ||
| 130 | 132 | ||
| 131 | #define RPC_IS_ASYNC(t) ((t)->tk_flags & RPC_TASK_ASYNC) | 133 | #define RPC_IS_ASYNC(t) ((t)->tk_flags & RPC_TASK_ASYNC) |
| 132 | #define RPC_IS_SWAPPER(t) ((t)->tk_flags & RPC_TASK_SWAPPER) | 134 | #define RPC_IS_SWAPPER(t) ((t)->tk_flags & RPC_TASK_SWAPPER) |
| 133 | #define RPC_DO_ROOTOVERRIDE(t) ((t)->tk_flags & RPC_TASK_ROOTCREDS) | 135 | #define RPC_DO_ROOTOVERRIDE(t) ((t)->tk_flags & RPC_TASK_ROOTCREDS) |
| 134 | #define RPC_ASSASSINATED(t) ((t)->tk_flags & RPC_TASK_KILLED) | 136 | #define RPC_ASSASSINATED(t) ((t)->tk_flags & RPC_TASK_KILLED) |
| 135 | #define RPC_IS_SOFT(t) ((t)->tk_flags & RPC_TASK_SOFT) | 137 | #define RPC_IS_SOFT(t) ((t)->tk_flags & (RPC_TASK_SOFT|RPC_TASK_TIMEOUT)) |
| 136 | #define RPC_IS_SOFTCONN(t) ((t)->tk_flags & RPC_TASK_SOFTCONN) | 138 | #define RPC_IS_SOFTCONN(t) ((t)->tk_flags & RPC_TASK_SOFTCONN) |
| 139 | #define RPC_WAS_SENT(t) ((t)->tk_flags & RPC_TASK_SENT) | ||
| 137 | 140 | ||
| 138 | #define RPC_TASK_RUNNING 0 | 141 | #define RPC_TASK_RUNNING 0 |
| 139 | #define RPC_TASK_QUEUED 1 | 142 | #define RPC_TASK_QUEUED 1 |
diff --git a/include/linux/usb/usbnet.h b/include/linux/usb/usbnet.h index 0e1855079fbb..605b0aa8d852 100644 --- a/include/linux/usb/usbnet.h +++ b/include/linux/usb/usbnet.h | |||
| @@ -68,6 +68,7 @@ struct usbnet { | |||
| 68 | # define EVENT_RX_PAUSED 5 | 68 | # define EVENT_RX_PAUSED 5 |
| 69 | # define EVENT_DEV_WAKING 6 | 69 | # define EVENT_DEV_WAKING 6 |
| 70 | # define EVENT_DEV_ASLEEP 7 | 70 | # define EVENT_DEV_ASLEEP 7 |
| 71 | # define EVENT_DEV_OPEN 8 | ||
| 71 | }; | 72 | }; |
| 72 | 73 | ||
| 73 | static inline struct usb_driver *driver_of(struct usb_interface *intf) | 74 | static inline struct usb_driver *driver_of(struct usb_interface *intf) |
diff --git a/include/linux/v4l2-mediabus.h b/include/linux/v4l2-mediabus.h index 7054a7a8065e..de5c15921025 100644 --- a/include/linux/v4l2-mediabus.h +++ b/include/linux/v4l2-mediabus.h | |||
| @@ -47,7 +47,7 @@ enum v4l2_mbus_pixelcode { | |||
| 47 | V4L2_MBUS_FMT_RGB565_2X8_BE = 0x1007, | 47 | V4L2_MBUS_FMT_RGB565_2X8_BE = 0x1007, |
| 48 | V4L2_MBUS_FMT_RGB565_2X8_LE = 0x1008, | 48 | V4L2_MBUS_FMT_RGB565_2X8_LE = 0x1008, |
| 49 | 49 | ||
| 50 | /* YUV (including grey) - next is 0x2013 */ | 50 | /* YUV (including grey) - next is 0x2014 */ |
| 51 | V4L2_MBUS_FMT_Y8_1X8 = 0x2001, | 51 | V4L2_MBUS_FMT_Y8_1X8 = 0x2001, |
| 52 | V4L2_MBUS_FMT_UYVY8_1_5X8 = 0x2002, | 52 | V4L2_MBUS_FMT_UYVY8_1_5X8 = 0x2002, |
| 53 | V4L2_MBUS_FMT_VYUY8_1_5X8 = 0x2003, | 53 | V4L2_MBUS_FMT_VYUY8_1_5X8 = 0x2003, |
| @@ -60,6 +60,7 @@ enum v4l2_mbus_pixelcode { | |||
| 60 | V4L2_MBUS_FMT_Y10_1X10 = 0x200a, | 60 | V4L2_MBUS_FMT_Y10_1X10 = 0x200a, |
| 61 | V4L2_MBUS_FMT_YUYV10_2X10 = 0x200b, | 61 | V4L2_MBUS_FMT_YUYV10_2X10 = 0x200b, |
| 62 | V4L2_MBUS_FMT_YVYU10_2X10 = 0x200c, | 62 | V4L2_MBUS_FMT_YVYU10_2X10 = 0x200c, |
| 63 | V4L2_MBUS_FMT_Y12_1X12 = 0x2013, | ||
| 63 | V4L2_MBUS_FMT_UYVY8_1X16 = 0x200f, | 64 | V4L2_MBUS_FMT_UYVY8_1X16 = 0x200f, |
| 64 | V4L2_MBUS_FMT_VYUY8_1X16 = 0x2010, | 65 | V4L2_MBUS_FMT_VYUY8_1X16 = 0x2010, |
| 65 | V4L2_MBUS_FMT_YUYV8_1X16 = 0x2011, | 66 | V4L2_MBUS_FMT_YUYV8_1X16 = 0x2011, |
| @@ -67,9 +68,11 @@ enum v4l2_mbus_pixelcode { | |||
| 67 | V4L2_MBUS_FMT_YUYV10_1X20 = 0x200d, | 68 | V4L2_MBUS_FMT_YUYV10_1X20 = 0x200d, |
| 68 | V4L2_MBUS_FMT_YVYU10_1X20 = 0x200e, | 69 | V4L2_MBUS_FMT_YVYU10_1X20 = 0x200e, |
| 69 | 70 | ||
| 70 | /* Bayer - next is 0x3013 */ | 71 | /* Bayer - next is 0x3015 */ |
| 71 | V4L2_MBUS_FMT_SBGGR8_1X8 = 0x3001, | 72 | V4L2_MBUS_FMT_SBGGR8_1X8 = 0x3001, |
| 73 | V4L2_MBUS_FMT_SGBRG8_1X8 = 0x3013, | ||
| 72 | V4L2_MBUS_FMT_SGRBG8_1X8 = 0x3002, | 74 | V4L2_MBUS_FMT_SGRBG8_1X8 = 0x3002, |
| 75 | V4L2_MBUS_FMT_SRGGB8_1X8 = 0x3014, | ||
| 73 | V4L2_MBUS_FMT_SBGGR10_DPCM8_1X8 = 0x300b, | 76 | V4L2_MBUS_FMT_SBGGR10_DPCM8_1X8 = 0x300b, |
| 74 | V4L2_MBUS_FMT_SGBRG10_DPCM8_1X8 = 0x300c, | 77 | V4L2_MBUS_FMT_SGBRG10_DPCM8_1X8 = 0x300c, |
| 75 | V4L2_MBUS_FMT_SGRBG10_DPCM8_1X8 = 0x3009, | 78 | V4L2_MBUS_FMT_SGRBG10_DPCM8_1X8 = 0x3009, |
diff --git a/include/linux/videodev2.h b/include/linux/videodev2.h index aa6c393b7ae9..be82c8ead1af 100644 --- a/include/linux/videodev2.h +++ b/include/linux/videodev2.h | |||
| @@ -308,6 +308,7 @@ struct v4l2_pix_format { | |||
| 308 | #define V4L2_PIX_FMT_Y4 v4l2_fourcc('Y', '0', '4', ' ') /* 4 Greyscale */ | 308 | #define V4L2_PIX_FMT_Y4 v4l2_fourcc('Y', '0', '4', ' ') /* 4 Greyscale */ |
| 309 | #define V4L2_PIX_FMT_Y6 v4l2_fourcc('Y', '0', '6', ' ') /* 6 Greyscale */ | 309 | #define V4L2_PIX_FMT_Y6 v4l2_fourcc('Y', '0', '6', ' ') /* 6 Greyscale */ |
| 310 | #define V4L2_PIX_FMT_Y10 v4l2_fourcc('Y', '1', '0', ' ') /* 10 Greyscale */ | 310 | #define V4L2_PIX_FMT_Y10 v4l2_fourcc('Y', '1', '0', ' ') /* 10 Greyscale */ |
| 311 | #define V4L2_PIX_FMT_Y12 v4l2_fourcc('Y', '1', '2', ' ') /* 12 Greyscale */ | ||
| 311 | #define V4L2_PIX_FMT_Y16 v4l2_fourcc('Y', '1', '6', ' ') /* 16 Greyscale */ | 312 | #define V4L2_PIX_FMT_Y16 v4l2_fourcc('Y', '1', '6', ' ') /* 16 Greyscale */ |
| 312 | 313 | ||
| 313 | /* Palette formats */ | 314 | /* Palette formats */ |
diff --git a/include/media/v4l2-device.h b/include/media/v4l2-device.h index bd102cf509ac..d61febfb1668 100644 --- a/include/media/v4l2-device.h +++ b/include/media/v4l2-device.h | |||
| @@ -163,7 +163,7 @@ v4l2_device_register_subdev_nodes(struct v4l2_device *v4l2_dev); | |||
| 163 | ({ \ | 163 | ({ \ |
| 164 | struct v4l2_subdev *__sd; \ | 164 | struct v4l2_subdev *__sd; \ |
| 165 | __v4l2_device_call_subdevs_until_err_p(v4l2_dev, __sd, cond, o, \ | 165 | __v4l2_device_call_subdevs_until_err_p(v4l2_dev, __sd, cond, o, \ |
| 166 | f, args...); \ | 166 | f , ##args); \ |
| 167 | }) | 167 | }) |
| 168 | 168 | ||
| 169 | /* Call the specified callback for all subdevs matching grp_id (if 0, then | 169 | /* Call the specified callback for all subdevs matching grp_id (if 0, then |
diff --git a/init/Kconfig b/init/Kconfig index 7a71e0a9992a..d886b1e9278e 100644 --- a/init/Kconfig +++ b/init/Kconfig | |||
| @@ -1226,6 +1226,7 @@ config SLAB | |||
| 1226 | per cpu and per node queues. | 1226 | per cpu and per node queues. |
| 1227 | 1227 | ||
| 1228 | config SLUB | 1228 | config SLUB |
| 1229 | depends on BROKEN || NUMA || !DISCONTIGMEM | ||
| 1229 | bool "SLUB (Unqueued Allocator)" | 1230 | bool "SLUB (Unqueued Allocator)" |
| 1230 | help | 1231 | help |
| 1231 | SLUB is a slab allocator that minimizes cache line usage | 1232 | SLUB is a slab allocator that minimizes cache line usage |
diff --git a/kernel/hrtimer.c b/kernel/hrtimer.c index 9017478c5d4c..87fdb3f8db14 100644 --- a/kernel/hrtimer.c +++ b/kernel/hrtimer.c | |||
| @@ -81,7 +81,11 @@ DEFINE_PER_CPU(struct hrtimer_cpu_base, hrtimer_bases) = | |||
| 81 | } | 81 | } |
| 82 | }; | 82 | }; |
| 83 | 83 | ||
| 84 | static int hrtimer_clock_to_base_table[MAX_CLOCKS]; | 84 | static int hrtimer_clock_to_base_table[MAX_CLOCKS] = { |
| 85 | [CLOCK_REALTIME] = HRTIMER_BASE_REALTIME, | ||
| 86 | [CLOCK_MONOTONIC] = HRTIMER_BASE_MONOTONIC, | ||
| 87 | [CLOCK_BOOTTIME] = HRTIMER_BASE_BOOTTIME, | ||
| 88 | }; | ||
| 85 | 89 | ||
| 86 | static inline int hrtimer_clockid_to_base(clockid_t clock_id) | 90 | static inline int hrtimer_clockid_to_base(clockid_t clock_id) |
| 87 | { | 91 | { |
| @@ -1722,10 +1726,6 @@ static struct notifier_block __cpuinitdata hrtimers_nb = { | |||
| 1722 | 1726 | ||
| 1723 | void __init hrtimers_init(void) | 1727 | void __init hrtimers_init(void) |
| 1724 | { | 1728 | { |
| 1725 | hrtimer_clock_to_base_table[CLOCK_REALTIME] = HRTIMER_BASE_REALTIME; | ||
| 1726 | hrtimer_clock_to_base_table[CLOCK_MONOTONIC] = HRTIMER_BASE_MONOTONIC; | ||
| 1727 | hrtimer_clock_to_base_table[CLOCK_BOOTTIME] = HRTIMER_BASE_BOOTTIME; | ||
| 1728 | |||
| 1729 | hrtimer_cpu_notify(&hrtimers_nb, (unsigned long)CPU_UP_PREPARE, | 1729 | hrtimer_cpu_notify(&hrtimers_nb, (unsigned long)CPU_UP_PREPARE, |
| 1730 | (void *)(long)smp_processor_id()); | 1730 | (void *)(long)smp_processor_id()); |
| 1731 | register_cpu_notifier(&hrtimers_nb); | 1731 | register_cpu_notifier(&hrtimers_nb); |
diff --git a/kernel/irq/proc.c b/kernel/irq/proc.c index dd201bd35103..834899f2500f 100644 --- a/kernel/irq/proc.c +++ b/kernel/irq/proc.c | |||
| @@ -419,7 +419,7 @@ int show_interrupts(struct seq_file *p, void *v) | |||
| 419 | } else { | 419 | } else { |
| 420 | seq_printf(p, " %8s", "None"); | 420 | seq_printf(p, " %8s", "None"); |
| 421 | } | 421 | } |
| 422 | #ifdef CONFIG_GENIRC_IRQ_SHOW_LEVEL | 422 | #ifdef CONFIG_GENERIC_IRQ_SHOW_LEVEL |
| 423 | seq_printf(p, " %-8s", irqd_is_level_type(&desc->irq_data) ? "Level" : "Edge"); | 423 | seq_printf(p, " %-8s", irqd_is_level_type(&desc->irq_data) ? "Level" : "Edge"); |
| 424 | #endif | 424 | #endif |
| 425 | if (desc->name) | 425 | if (desc->name) |
diff --git a/kernel/trace/Kconfig b/kernel/trace/Kconfig index 61d7d59f4a1a..2ad39e556cb4 100644 --- a/kernel/trace/Kconfig +++ b/kernel/trace/Kconfig | |||
| @@ -141,7 +141,7 @@ if FTRACE | |||
| 141 | config FUNCTION_TRACER | 141 | config FUNCTION_TRACER |
| 142 | bool "Kernel Function Tracer" | 142 | bool "Kernel Function Tracer" |
| 143 | depends on HAVE_FUNCTION_TRACER | 143 | depends on HAVE_FUNCTION_TRACER |
| 144 | select FRAME_POINTER if !ARM_UNWIND && !S390 | 144 | select FRAME_POINTER if !ARM_UNWIND && !S390 && !MICROBLAZE |
| 145 | select KALLSYMS | 145 | select KALLSYMS |
| 146 | select GENERIC_TRACER | 146 | select GENERIC_TRACER |
| 147 | select CONTEXT_SWITCH_TRACER | 147 | select CONTEXT_SWITCH_TRACER |
diff --git a/kernel/watchdog.c b/kernel/watchdog.c index 140dce750450..14733d4d156b 100644 --- a/kernel/watchdog.c +++ b/kernel/watchdog.c | |||
| @@ -430,9 +430,12 @@ static int watchdog_enable(int cpu) | |||
| 430 | p = kthread_create(watchdog, (void *)(unsigned long)cpu, "watchdog/%d", cpu); | 430 | p = kthread_create(watchdog, (void *)(unsigned long)cpu, "watchdog/%d", cpu); |
| 431 | if (IS_ERR(p)) { | 431 | if (IS_ERR(p)) { |
| 432 | printk(KERN_ERR "softlockup watchdog for %i failed\n", cpu); | 432 | printk(KERN_ERR "softlockup watchdog for %i failed\n", cpu); |
| 433 | if (!err) | 433 | if (!err) { |
| 434 | /* if hardlockup hasn't already set this */ | 434 | /* if hardlockup hasn't already set this */ |
| 435 | err = PTR_ERR(p); | 435 | err = PTR_ERR(p); |
| 436 | /* and disable the perf event */ | ||
| 437 | watchdog_nmi_disable(cpu); | ||
| 438 | } | ||
| 436 | goto out; | 439 | goto out; |
| 437 | } | 440 | } |
| 438 | kthread_bind(p, cpu); | 441 | kthread_bind(p, cpu); |
diff --git a/kernel/workqueue.c b/kernel/workqueue.c index 8859a41806dd..e3378e8d3a5c 100644 --- a/kernel/workqueue.c +++ b/kernel/workqueue.c | |||
| @@ -1291,8 +1291,14 @@ __acquires(&gcwq->lock) | |||
| 1291 | return true; | 1291 | return true; |
| 1292 | spin_unlock_irq(&gcwq->lock); | 1292 | spin_unlock_irq(&gcwq->lock); |
| 1293 | 1293 | ||
| 1294 | /* CPU has come up in between, retry migration */ | 1294 | /* |
| 1295 | * We've raced with CPU hot[un]plug. Give it a breather | ||
| 1296 | * and retry migration. cond_resched() is required here; | ||
| 1297 | * otherwise, we might deadlock against cpu_stop trying to | ||
| 1298 | * bring down the CPU on non-preemptive kernel. | ||
| 1299 | */ | ||
| 1295 | cpu_relax(); | 1300 | cpu_relax(); |
| 1301 | cond_resched(); | ||
| 1296 | } | 1302 | } |
| 1297 | } | 1303 | } |
| 1298 | 1304 | ||
diff --git a/lib/xz/xz_dec_lzma2.c b/lib/xz/xz_dec_lzma2.c index ea5fa4fe9d67..a6cdc969ea42 100644 --- a/lib/xz/xz_dec_lzma2.c +++ b/lib/xz/xz_dec_lzma2.c | |||
| @@ -969,6 +969,9 @@ XZ_EXTERN enum xz_ret xz_dec_lzma2_run(struct xz_dec_lzma2 *s, | |||
| 969 | */ | 969 | */ |
| 970 | tmp = b->in[b->in_pos++]; | 970 | tmp = b->in[b->in_pos++]; |
| 971 | 971 | ||
| 972 | if (tmp == 0x00) | ||
| 973 | return XZ_STREAM_END; | ||
| 974 | |||
| 972 | if (tmp >= 0xE0 || tmp == 0x01) { | 975 | if (tmp >= 0xE0 || tmp == 0x01) { |
| 973 | s->lzma2.need_props = true; | 976 | s->lzma2.need_props = true; |
| 974 | s->lzma2.need_dict_reset = false; | 977 | s->lzma2.need_dict_reset = false; |
| @@ -1001,9 +1004,6 @@ XZ_EXTERN enum xz_ret xz_dec_lzma2_run(struct xz_dec_lzma2 *s, | |||
| 1001 | lzma_reset(s); | 1004 | lzma_reset(s); |
| 1002 | } | 1005 | } |
| 1003 | } else { | 1006 | } else { |
| 1004 | if (tmp == 0x00) | ||
| 1005 | return XZ_STREAM_END; | ||
| 1006 | |||
| 1007 | if (tmp > 0x02) | 1007 | if (tmp > 0x02) |
| 1008 | return XZ_DATA_ERROR; | 1008 | return XZ_DATA_ERROR; |
| 1009 | 1009 | ||
diff --git a/mm/huge_memory.c b/mm/huge_memory.c index 470dcda10add..83326ad66d9b 100644 --- a/mm/huge_memory.c +++ b/mm/huge_memory.c | |||
| @@ -1408,6 +1408,9 @@ out: | |||
| 1408 | return ret; | 1408 | return ret; |
| 1409 | } | 1409 | } |
| 1410 | 1410 | ||
| 1411 | #define VM_NO_THP (VM_SPECIAL|VM_INSERTPAGE|VM_MIXEDMAP|VM_SAO| \ | ||
| 1412 | VM_HUGETLB|VM_SHARED|VM_MAYSHARE) | ||
| 1413 | |||
| 1411 | int hugepage_madvise(struct vm_area_struct *vma, | 1414 | int hugepage_madvise(struct vm_area_struct *vma, |
| 1412 | unsigned long *vm_flags, int advice) | 1415 | unsigned long *vm_flags, int advice) |
| 1413 | { | 1416 | { |
| @@ -1416,11 +1419,7 @@ int hugepage_madvise(struct vm_area_struct *vma, | |||
| 1416 | /* | 1419 | /* |
| 1417 | * Be somewhat over-protective like KSM for now! | 1420 | * Be somewhat over-protective like KSM for now! |
| 1418 | */ | 1421 | */ |
| 1419 | if (*vm_flags & (VM_HUGEPAGE | | 1422 | if (*vm_flags & (VM_HUGEPAGE | VM_NO_THP)) |
| 1420 | VM_SHARED | VM_MAYSHARE | | ||
| 1421 | VM_PFNMAP | VM_IO | VM_DONTEXPAND | | ||
| 1422 | VM_RESERVED | VM_HUGETLB | VM_INSERTPAGE | | ||
| 1423 | VM_MIXEDMAP | VM_SAO)) | ||
| 1424 | return -EINVAL; | 1423 | return -EINVAL; |
| 1425 | *vm_flags &= ~VM_NOHUGEPAGE; | 1424 | *vm_flags &= ~VM_NOHUGEPAGE; |
| 1426 | *vm_flags |= VM_HUGEPAGE; | 1425 | *vm_flags |= VM_HUGEPAGE; |
| @@ -1436,11 +1435,7 @@ int hugepage_madvise(struct vm_area_struct *vma, | |||
| 1436 | /* | 1435 | /* |
| 1437 | * Be somewhat over-protective like KSM for now! | 1436 | * Be somewhat over-protective like KSM for now! |
| 1438 | */ | 1437 | */ |
| 1439 | if (*vm_flags & (VM_NOHUGEPAGE | | 1438 | if (*vm_flags & (VM_NOHUGEPAGE | VM_NO_THP)) |
| 1440 | VM_SHARED | VM_MAYSHARE | | ||
| 1441 | VM_PFNMAP | VM_IO | VM_DONTEXPAND | | ||
| 1442 | VM_RESERVED | VM_HUGETLB | VM_INSERTPAGE | | ||
| 1443 | VM_MIXEDMAP | VM_SAO)) | ||
| 1444 | return -EINVAL; | 1439 | return -EINVAL; |
| 1445 | *vm_flags &= ~VM_HUGEPAGE; | 1440 | *vm_flags &= ~VM_HUGEPAGE; |
| 1446 | *vm_flags |= VM_NOHUGEPAGE; | 1441 | *vm_flags |= VM_NOHUGEPAGE; |
| @@ -1574,10 +1569,14 @@ int khugepaged_enter_vma_merge(struct vm_area_struct *vma) | |||
| 1574 | * page fault if needed. | 1569 | * page fault if needed. |
| 1575 | */ | 1570 | */ |
| 1576 | return 0; | 1571 | return 0; |
| 1577 | if (vma->vm_file || vma->vm_ops) | 1572 | if (vma->vm_ops) |
| 1578 | /* khugepaged not yet working on file or special mappings */ | 1573 | /* khugepaged not yet working on file or special mappings */ |
| 1579 | return 0; | 1574 | return 0; |
| 1580 | VM_BUG_ON(is_linear_pfn_mapping(vma) || is_pfn_mapping(vma)); | 1575 | /* |
| 1576 | * If is_pfn_mapping() is true is_learn_pfn_mapping() must be | ||
| 1577 | * true too, verify it here. | ||
| 1578 | */ | ||
| 1579 | VM_BUG_ON(is_linear_pfn_mapping(vma) || vma->vm_flags & VM_NO_THP); | ||
| 1581 | hstart = (vma->vm_start + ~HPAGE_PMD_MASK) & HPAGE_PMD_MASK; | 1580 | hstart = (vma->vm_start + ~HPAGE_PMD_MASK) & HPAGE_PMD_MASK; |
| 1582 | hend = vma->vm_end & HPAGE_PMD_MASK; | 1581 | hend = vma->vm_end & HPAGE_PMD_MASK; |
| 1583 | if (hstart < hend) | 1582 | if (hstart < hend) |
| @@ -1828,12 +1827,15 @@ static void collapse_huge_page(struct mm_struct *mm, | |||
| 1828 | (vma->vm_flags & VM_NOHUGEPAGE)) | 1827 | (vma->vm_flags & VM_NOHUGEPAGE)) |
| 1829 | goto out; | 1828 | goto out; |
| 1830 | 1829 | ||
| 1831 | /* VM_PFNMAP vmas may have vm_ops null but vm_file set */ | 1830 | if (!vma->anon_vma || vma->vm_ops) |
| 1832 | if (!vma->anon_vma || vma->vm_ops || vma->vm_file) | ||
| 1833 | goto out; | 1831 | goto out; |
| 1834 | if (is_vma_temporary_stack(vma)) | 1832 | if (is_vma_temporary_stack(vma)) |
| 1835 | goto out; | 1833 | goto out; |
| 1836 | VM_BUG_ON(is_linear_pfn_mapping(vma) || is_pfn_mapping(vma)); | 1834 | /* |
| 1835 | * If is_pfn_mapping() is true is_learn_pfn_mapping() must be | ||
| 1836 | * true too, verify it here. | ||
| 1837 | */ | ||
| 1838 | VM_BUG_ON(is_linear_pfn_mapping(vma) || vma->vm_flags & VM_NO_THP); | ||
| 1837 | 1839 | ||
| 1838 | pgd = pgd_offset(mm, address); | 1840 | pgd = pgd_offset(mm, address); |
| 1839 | if (!pgd_present(*pgd)) | 1841 | if (!pgd_present(*pgd)) |
| @@ -2066,13 +2068,16 @@ static unsigned int khugepaged_scan_mm_slot(unsigned int pages, | |||
| 2066 | progress++; | 2068 | progress++; |
| 2067 | continue; | 2069 | continue; |
| 2068 | } | 2070 | } |
| 2069 | /* VM_PFNMAP vmas may have vm_ops null but vm_file set */ | 2071 | if (!vma->anon_vma || vma->vm_ops) |
| 2070 | if (!vma->anon_vma || vma->vm_ops || vma->vm_file) | ||
| 2071 | goto skip; | 2072 | goto skip; |
| 2072 | if (is_vma_temporary_stack(vma)) | 2073 | if (is_vma_temporary_stack(vma)) |
| 2073 | goto skip; | 2074 | goto skip; |
| 2074 | 2075 | /* | |
| 2075 | VM_BUG_ON(is_linear_pfn_mapping(vma) || is_pfn_mapping(vma)); | 2076 | * If is_pfn_mapping() is true is_learn_pfn_mapping() |
| 2077 | * must be true too, verify it here. | ||
| 2078 | */ | ||
| 2079 | VM_BUG_ON(is_linear_pfn_mapping(vma) || | ||
| 2080 | vma->vm_flags & VM_NO_THP); | ||
| 2076 | 2081 | ||
| 2077 | hstart = (vma->vm_start + ~HPAGE_PMD_MASK) & HPAGE_PMD_MASK; | 2082 | hstart = (vma->vm_start + ~HPAGE_PMD_MASK) & HPAGE_PMD_MASK; |
| 2078 | hend = vma->vm_end & HPAGE_PMD_MASK; | 2083 | hend = vma->vm_end & HPAGE_PMD_MASK; |
diff --git a/mm/memory.c b/mm/memory.c index ce22a250926f..607098d47e74 100644 --- a/mm/memory.c +++ b/mm/memory.c | |||
| @@ -3396,7 +3396,7 @@ int handle_mm_fault(struct mm_struct *mm, struct vm_area_struct *vma, | |||
| 3396 | * run pte_offset_map on the pmd, if an huge pmd could | 3396 | * run pte_offset_map on the pmd, if an huge pmd could |
| 3397 | * materialize from under us from a different thread. | 3397 | * materialize from under us from a different thread. |
| 3398 | */ | 3398 | */ |
| 3399 | if (unlikely(__pte_alloc(mm, vma, pmd, address))) | 3399 | if (unlikely(pmd_none(*pmd)) && __pte_alloc(mm, vma, pmd, address)) |
| 3400 | return VM_FAULT_OOM; | 3400 | return VM_FAULT_OOM; |
| 3401 | /* if an huge pmd materialized from under us just retry later */ | 3401 | /* if an huge pmd materialized from under us just retry later */ |
| 3402 | if (unlikely(pmd_trans_huge(*pmd))) | 3402 | if (unlikely(pmd_trans_huge(*pmd))) |
diff --git a/mm/oom_kill.c b/mm/oom_kill.c index 83fb72c108b7..f52e85c80e8d 100644 --- a/mm/oom_kill.c +++ b/mm/oom_kill.c | |||
| @@ -172,10 +172,13 @@ unsigned int oom_badness(struct task_struct *p, struct mem_cgroup *mem, | |||
| 172 | 172 | ||
| 173 | /* | 173 | /* |
| 174 | * The baseline for the badness score is the proportion of RAM that each | 174 | * The baseline for the badness score is the proportion of RAM that each |
| 175 | * task's rss and swap space use. | 175 | * task's rss, pagetable and swap space use. |
| 176 | */ | 176 | */ |
| 177 | points = (get_mm_rss(p->mm) + get_mm_counter(p->mm, MM_SWAPENTS)) * 1000 / | 177 | points = get_mm_rss(p->mm) + p->mm->nr_ptes; |
| 178 | totalpages; | 178 | points += get_mm_counter(p->mm, MM_SWAPENTS); |
| 179 | |||
| 180 | points *= 1000; | ||
| 181 | points /= totalpages; | ||
| 179 | task_unlock(p); | 182 | task_unlock(p); |
| 180 | 183 | ||
| 181 | /* | 184 | /* |
diff --git a/net/bluetooth/hci_core.c b/net/bluetooth/hci_core.c index c83f618282f7..b5a8afc2be33 100644 --- a/net/bluetooth/hci_core.c +++ b/net/bluetooth/hci_core.c | |||
| @@ -587,10 +587,8 @@ static int hci_dev_do_close(struct hci_dev *hdev) | |||
| 587 | hci_req_cancel(hdev, ENODEV); | 587 | hci_req_cancel(hdev, ENODEV); |
| 588 | hci_req_lock(hdev); | 588 | hci_req_lock(hdev); |
| 589 | 589 | ||
| 590 | /* Stop timer, it might be running */ | ||
| 591 | del_timer_sync(&hdev->cmd_timer); | ||
| 592 | |||
| 593 | if (!test_and_clear_bit(HCI_UP, &hdev->flags)) { | 590 | if (!test_and_clear_bit(HCI_UP, &hdev->flags)) { |
| 591 | del_timer_sync(&hdev->cmd_timer); | ||
| 594 | hci_req_unlock(hdev); | 592 | hci_req_unlock(hdev); |
| 595 | return 0; | 593 | return 0; |
| 596 | } | 594 | } |
| @@ -629,6 +627,7 @@ static int hci_dev_do_close(struct hci_dev *hdev) | |||
| 629 | 627 | ||
| 630 | /* Drop last sent command */ | 628 | /* Drop last sent command */ |
| 631 | if (hdev->sent_cmd) { | 629 | if (hdev->sent_cmd) { |
| 630 | del_timer_sync(&hdev->cmd_timer); | ||
| 632 | kfree_skb(hdev->sent_cmd); | 631 | kfree_skb(hdev->sent_cmd); |
| 633 | hdev->sent_cmd = NULL; | 632 | hdev->sent_cmd = NULL; |
| 634 | } | 633 | } |
diff --git a/net/bluetooth/hci_event.c b/net/bluetooth/hci_event.c index cebe7588469f..b2570159a044 100644 --- a/net/bluetooth/hci_event.c +++ b/net/bluetooth/hci_event.c | |||
| @@ -2387,8 +2387,6 @@ static inline void hci_io_capa_reply_evt(struct hci_dev *hdev, struct sk_buff *s | |||
| 2387 | if (!conn) | 2387 | if (!conn) |
| 2388 | goto unlock; | 2388 | goto unlock; |
| 2389 | 2389 | ||
| 2390 | hci_conn_hold(conn); | ||
| 2391 | |||
| 2392 | conn->remote_cap = ev->capability; | 2390 | conn->remote_cap = ev->capability; |
| 2393 | conn->remote_oob = ev->oob_data; | 2391 | conn->remote_oob = ev->oob_data; |
| 2394 | conn->remote_auth = ev->authentication; | 2392 | conn->remote_auth = ev->authentication; |
diff --git a/net/bluetooth/l2cap_core.c b/net/bluetooth/l2cap_core.c index ca27f3a41536..2c8dd4494c63 100644 --- a/net/bluetooth/l2cap_core.c +++ b/net/bluetooth/l2cap_core.c | |||
| @@ -1051,6 +1051,7 @@ static void l2cap_retransmit_one_frame(struct sock *sk, u8 tx_seq) | |||
| 1051 | tx_skb = skb_clone(skb, GFP_ATOMIC); | 1051 | tx_skb = skb_clone(skb, GFP_ATOMIC); |
| 1052 | bt_cb(skb)->retries++; | 1052 | bt_cb(skb)->retries++; |
| 1053 | control = get_unaligned_le16(tx_skb->data + L2CAP_HDR_SIZE); | 1053 | control = get_unaligned_le16(tx_skb->data + L2CAP_HDR_SIZE); |
| 1054 | control &= L2CAP_CTRL_SAR; | ||
| 1054 | 1055 | ||
| 1055 | if (pi->conn_state & L2CAP_CONN_SEND_FBIT) { | 1056 | if (pi->conn_state & L2CAP_CONN_SEND_FBIT) { |
| 1056 | control |= L2CAP_CTRL_FINAL; | 1057 | control |= L2CAP_CTRL_FINAL; |
diff --git a/net/bluetooth/sco.c b/net/bluetooth/sco.c index 42fdffd1d76c..94954c74f6ae 100644 --- a/net/bluetooth/sco.c +++ b/net/bluetooth/sco.c | |||
| @@ -369,6 +369,15 @@ static void __sco_sock_close(struct sock *sk) | |||
| 369 | 369 | ||
| 370 | case BT_CONNECTED: | 370 | case BT_CONNECTED: |
| 371 | case BT_CONFIG: | 371 | case BT_CONFIG: |
| 372 | if (sco_pi(sk)->conn) { | ||
| 373 | sk->sk_state = BT_DISCONN; | ||
| 374 | sco_sock_set_timer(sk, SCO_DISCONN_TIMEOUT); | ||
| 375 | hci_conn_put(sco_pi(sk)->conn->hcon); | ||
| 376 | sco_pi(sk)->conn = NULL; | ||
| 377 | } else | ||
| 378 | sco_chan_del(sk, ECONNRESET); | ||
| 379 | break; | ||
| 380 | |||
| 372 | case BT_CONNECT: | 381 | case BT_CONNECT: |
| 373 | case BT_DISCONN: | 382 | case BT_DISCONN: |
| 374 | sco_chan_del(sk, ECONNRESET); | 383 | sco_chan_del(sk, ECONNRESET); |
diff --git a/net/bridge/br_input.c b/net/bridge/br_input.c index e2160792e1bc..0c7badad62af 100644 --- a/net/bridge/br_input.c +++ b/net/bridge/br_input.c | |||
| @@ -164,7 +164,7 @@ rx_handler_result_t br_handle_frame(struct sk_buff **pskb) | |||
| 164 | goto drop; | 164 | goto drop; |
| 165 | 165 | ||
| 166 | /* If STP is turned off, then forward */ | 166 | /* If STP is turned off, then forward */ |
| 167 | if (p->br->stp_enabled == BR_NO_STP) | 167 | if (p->br->stp_enabled == BR_NO_STP && dest[5] == 0) |
| 168 | goto forward; | 168 | goto forward; |
| 169 | 169 | ||
| 170 | if (NF_HOOK(NFPROTO_BRIDGE, NF_BR_LOCAL_IN, skb, skb->dev, | 170 | if (NF_HOOK(NFPROTO_BRIDGE, NF_BR_LOCAL_IN, skb, skb->dev, |
diff --git a/net/can/bcm.c b/net/can/bcm.c index 57b1aed79014..8a6a05e7c3c8 100644 --- a/net/can/bcm.c +++ b/net/can/bcm.c | |||
| @@ -1427,9 +1427,14 @@ static int bcm_init(struct sock *sk) | |||
| 1427 | static int bcm_release(struct socket *sock) | 1427 | static int bcm_release(struct socket *sock) |
| 1428 | { | 1428 | { |
| 1429 | struct sock *sk = sock->sk; | 1429 | struct sock *sk = sock->sk; |
| 1430 | struct bcm_sock *bo = bcm_sk(sk); | 1430 | struct bcm_sock *bo; |
| 1431 | struct bcm_op *op, *next; | 1431 | struct bcm_op *op, *next; |
| 1432 | 1432 | ||
| 1433 | if (sk == NULL) | ||
| 1434 | return 0; | ||
| 1435 | |||
| 1436 | bo = bcm_sk(sk); | ||
| 1437 | |||
| 1433 | /* remove bcm_ops, timer, rx_unregister(), etc. */ | 1438 | /* remove bcm_ops, timer, rx_unregister(), etc. */ |
| 1434 | 1439 | ||
| 1435 | unregister_netdevice_notifier(&bo->notifier); | 1440 | unregister_netdevice_notifier(&bo->notifier); |
diff --git a/net/can/raw.c b/net/can/raw.c index 649acfa7c70a..0eb39a7fdf64 100644 --- a/net/can/raw.c +++ b/net/can/raw.c | |||
| @@ -305,7 +305,12 @@ static int raw_init(struct sock *sk) | |||
| 305 | static int raw_release(struct socket *sock) | 305 | static int raw_release(struct socket *sock) |
| 306 | { | 306 | { |
| 307 | struct sock *sk = sock->sk; | 307 | struct sock *sk = sock->sk; |
| 308 | struct raw_sock *ro = raw_sk(sk); | 308 | struct raw_sock *ro; |
| 309 | |||
| 310 | if (!sk) | ||
| 311 | return 0; | ||
| 312 | |||
| 313 | ro = raw_sk(sk); | ||
| 309 | 314 | ||
| 310 | unregister_netdevice_notifier(&ro->notifier); | 315 | unregister_netdevice_notifier(&ro->notifier); |
| 311 | 316 | ||
diff --git a/net/core/dev.c b/net/core/dev.c index c2ac599fa0f6..856b6ee9a1d5 100644 --- a/net/core/dev.c +++ b/net/core/dev.c | |||
| @@ -4773,7 +4773,7 @@ static int dev_ifsioc_locked(struct net *net, struct ifreq *ifr, unsigned int cm | |||
| 4773 | * is never reached | 4773 | * is never reached |
| 4774 | */ | 4774 | */ |
| 4775 | WARN_ON(1); | 4775 | WARN_ON(1); |
| 4776 | err = -EINVAL; | 4776 | err = -ENOTTY; |
| 4777 | break; | 4777 | break; |
| 4778 | 4778 | ||
| 4779 | } | 4779 | } |
| @@ -5041,7 +5041,7 @@ int dev_ioctl(struct net *net, unsigned int cmd, void __user *arg) | |||
| 5041 | /* Set the per device memory buffer space. | 5041 | /* Set the per device memory buffer space. |
| 5042 | * Not applicable in our case */ | 5042 | * Not applicable in our case */ |
| 5043 | case SIOCSIFLINK: | 5043 | case SIOCSIFLINK: |
| 5044 | return -EINVAL; | 5044 | return -ENOTTY; |
| 5045 | 5045 | ||
| 5046 | /* | 5046 | /* |
| 5047 | * Unknown or private ioctl. | 5047 | * Unknown or private ioctl. |
| @@ -5062,7 +5062,7 @@ int dev_ioctl(struct net *net, unsigned int cmd, void __user *arg) | |||
| 5062 | /* Take care of Wireless Extensions */ | 5062 | /* Take care of Wireless Extensions */ |
| 5063 | if (cmd >= SIOCIWFIRST && cmd <= SIOCIWLAST) | 5063 | if (cmd >= SIOCIWFIRST && cmd <= SIOCIWLAST) |
| 5064 | return wext_handle_ioctl(net, &ifr, cmd, arg); | 5064 | return wext_handle_ioctl(net, &ifr, cmd, arg); |
| 5065 | return -EINVAL; | 5065 | return -ENOTTY; |
| 5066 | } | 5066 | } |
| 5067 | } | 5067 | } |
| 5068 | 5068 | ||
diff --git a/net/dsa/Kconfig b/net/dsa/Kconfig index 87bb5f4de0e8..c53ded2a98df 100644 --- a/net/dsa/Kconfig +++ b/net/dsa/Kconfig | |||
| @@ -41,12 +41,12 @@ config NET_DSA_MV88E6XXX_NEED_PPU | |||
| 41 | default n | 41 | default n |
| 42 | 42 | ||
| 43 | config NET_DSA_MV88E6131 | 43 | config NET_DSA_MV88E6131 |
| 44 | bool "Marvell 88E6095/6095F/6131 ethernet switch chip support" | 44 | bool "Marvell 88E6085/6095/6095F/6131 ethernet switch chip support" |
| 45 | select NET_DSA_MV88E6XXX | 45 | select NET_DSA_MV88E6XXX |
| 46 | select NET_DSA_MV88E6XXX_NEED_PPU | 46 | select NET_DSA_MV88E6XXX_NEED_PPU |
| 47 | select NET_DSA_TAG_DSA | 47 | select NET_DSA_TAG_DSA |
| 48 | ---help--- | 48 | ---help--- |
| 49 | This enables support for the Marvell 88E6095/6095F/6131 | 49 | This enables support for the Marvell 88E6085/6095/6095F/6131 |
| 50 | ethernet switch chips. | 50 | ethernet switch chips. |
| 51 | 51 | ||
| 52 | config NET_DSA_MV88E6123_61_65 | 52 | config NET_DSA_MV88E6123_61_65 |
diff --git a/net/dsa/mv88e6131.c b/net/dsa/mv88e6131.c index 3da418894efc..45f7411e90ba 100644 --- a/net/dsa/mv88e6131.c +++ b/net/dsa/mv88e6131.c | |||
| @@ -207,8 +207,15 @@ static int mv88e6131_setup_port(struct dsa_switch *ds, int p) | |||
| 207 | * mode, but do not enable forwarding of unknown unicasts. | 207 | * mode, but do not enable forwarding of unknown unicasts. |
| 208 | */ | 208 | */ |
| 209 | val = 0x0433; | 209 | val = 0x0433; |
| 210 | if (p == dsa_upstream_port(ds)) | 210 | if (p == dsa_upstream_port(ds)) { |
| 211 | val |= 0x0104; | 211 | val |= 0x0104; |
| 212 | /* | ||
| 213 | * On 6085, unknown multicast forward is controlled | ||
| 214 | * here rather than in Port Control 2 register. | ||
| 215 | */ | ||
| 216 | if (ps->id == ID_6085) | ||
| 217 | val |= 0x0008; | ||
| 218 | } | ||
| 212 | if (ds->dsa_port_mask & (1 << p)) | 219 | if (ds->dsa_port_mask & (1 << p)) |
| 213 | val |= 0x0100; | 220 | val |= 0x0100; |
| 214 | REG_WRITE(addr, 0x04, val); | 221 | REG_WRITE(addr, 0x04, val); |
| @@ -251,10 +258,19 @@ static int mv88e6131_setup_port(struct dsa_switch *ds, int p) | |||
| 251 | * If this is the upstream port for this switch, enable | 258 | * If this is the upstream port for this switch, enable |
| 252 | * forwarding of unknown multicast addresses. | 259 | * forwarding of unknown multicast addresses. |
| 253 | */ | 260 | */ |
| 254 | val = 0x0080 | dsa_upstream_port(ds); | 261 | if (ps->id == ID_6085) |
| 255 | if (p == dsa_upstream_port(ds)) | 262 | /* |
| 256 | val |= 0x0040; | 263 | * on 6085, bits 3:0 are reserved, bit 6 control ARP |
| 257 | REG_WRITE(addr, 0x08, val); | 264 | * mirroring, and multicast forward is handled in |
| 265 | * Port Control register. | ||
| 266 | */ | ||
| 267 | REG_WRITE(addr, 0x08, 0x0080); | ||
| 268 | else { | ||
| 269 | val = 0x0080 | dsa_upstream_port(ds); | ||
| 270 | if (p == dsa_upstream_port(ds)) | ||
| 271 | val |= 0x0040; | ||
| 272 | REG_WRITE(addr, 0x08, val); | ||
| 273 | } | ||
| 258 | 274 | ||
| 259 | /* | 275 | /* |
| 260 | * Rate Control: disable ingress rate limiting. | 276 | * Rate Control: disable ingress rate limiting. |
diff --git a/net/ipv4/devinet.c b/net/ipv4/devinet.c index 5345b0bee6df..cd9ca0811cfa 100644 --- a/net/ipv4/devinet.c +++ b/net/ipv4/devinet.c | |||
| @@ -1680,7 +1680,7 @@ static void __devinet_sysctl_unregister(struct ipv4_devconf *cnf) | |||
| 1680 | return; | 1680 | return; |
| 1681 | 1681 | ||
| 1682 | cnf->sysctl = NULL; | 1682 | cnf->sysctl = NULL; |
| 1683 | unregister_sysctl_table(t->sysctl_header); | 1683 | unregister_net_sysctl_table(t->sysctl_header); |
| 1684 | kfree(t->dev_name); | 1684 | kfree(t->dev_name); |
| 1685 | kfree(t); | 1685 | kfree(t); |
| 1686 | } | 1686 | } |
diff --git a/net/ipv4/fib_trie.c b/net/ipv4/fib_trie.c index e9013d6c1f51..5fe9b8b41df3 100644 --- a/net/ipv4/fib_trie.c +++ b/net/ipv4/fib_trie.c | |||
| @@ -1978,9 +1978,6 @@ struct fib_table *fib_trie_table(u32 id) | |||
| 1978 | t = (struct trie *) tb->tb_data; | 1978 | t = (struct trie *) tb->tb_data; |
| 1979 | memset(t, 0, sizeof(*t)); | 1979 | memset(t, 0, sizeof(*t)); |
| 1980 | 1980 | ||
| 1981 | if (id == RT_TABLE_LOCAL) | ||
| 1982 | pr_info("IPv4 FIB: Using LC-trie version %s\n", VERSION); | ||
| 1983 | |||
| 1984 | return tb; | 1981 | return tb; |
| 1985 | } | 1982 | } |
| 1986 | 1983 | ||
diff --git a/net/ipv4/route.c b/net/ipv4/route.c index c1acf69858fd..99e6e4bb1c72 100644 --- a/net/ipv4/route.c +++ b/net/ipv4/route.c | |||
| @@ -2690,6 +2690,12 @@ static void ipv4_rt_blackhole_update_pmtu(struct dst_entry *dst, u32 mtu) | |||
| 2690 | { | 2690 | { |
| 2691 | } | 2691 | } |
| 2692 | 2692 | ||
| 2693 | static u32 *ipv4_rt_blackhole_cow_metrics(struct dst_entry *dst, | ||
| 2694 | unsigned long old) | ||
| 2695 | { | ||
| 2696 | return NULL; | ||
| 2697 | } | ||
| 2698 | |||
| 2693 | static struct dst_ops ipv4_dst_blackhole_ops = { | 2699 | static struct dst_ops ipv4_dst_blackhole_ops = { |
| 2694 | .family = AF_INET, | 2700 | .family = AF_INET, |
| 2695 | .protocol = cpu_to_be16(ETH_P_IP), | 2701 | .protocol = cpu_to_be16(ETH_P_IP), |
| @@ -2698,6 +2704,7 @@ static struct dst_ops ipv4_dst_blackhole_ops = { | |||
| 2698 | .default_mtu = ipv4_blackhole_default_mtu, | 2704 | .default_mtu = ipv4_blackhole_default_mtu, |
| 2699 | .default_advmss = ipv4_default_advmss, | 2705 | .default_advmss = ipv4_default_advmss, |
| 2700 | .update_pmtu = ipv4_rt_blackhole_update_pmtu, | 2706 | .update_pmtu = ipv4_rt_blackhole_update_pmtu, |
| 2707 | .cow_metrics = ipv4_rt_blackhole_cow_metrics, | ||
| 2701 | }; | 2708 | }; |
| 2702 | 2709 | ||
| 2703 | struct dst_entry *ipv4_blackhole_route(struct net *net, struct dst_entry *dst_orig) | 2710 | struct dst_entry *ipv4_blackhole_route(struct net *net, struct dst_entry *dst_orig) |
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c index 1493534116df..a7bda0757053 100644 --- a/net/ipv6/addrconf.c +++ b/net/ipv6/addrconf.c | |||
| @@ -4537,7 +4537,7 @@ static void __addrconf_sysctl_unregister(struct ipv6_devconf *p) | |||
| 4537 | 4537 | ||
| 4538 | t = p->sysctl; | 4538 | t = p->sysctl; |
| 4539 | p->sysctl = NULL; | 4539 | p->sysctl = NULL; |
| 4540 | unregister_sysctl_table(t->sysctl_header); | 4540 | unregister_net_sysctl_table(t->sysctl_header); |
| 4541 | kfree(t->dev_name); | 4541 | kfree(t->dev_name); |
| 4542 | kfree(t); | 4542 | kfree(t); |
| 4543 | } | 4543 | } |
diff --git a/net/ipv6/esp6.c b/net/ipv6/esp6.c index 5aa8ec88f194..59dccfbb5b11 100644 --- a/net/ipv6/esp6.c +++ b/net/ipv6/esp6.c | |||
| @@ -371,7 +371,7 @@ static int esp6_input(struct xfrm_state *x, struct sk_buff *skb) | |||
| 371 | iv = esp_tmp_iv(aead, tmp, seqhilen); | 371 | iv = esp_tmp_iv(aead, tmp, seqhilen); |
| 372 | req = esp_tmp_req(aead, iv); | 372 | req = esp_tmp_req(aead, iv); |
| 373 | asg = esp_req_sg(aead, req); | 373 | asg = esp_req_sg(aead, req); |
| 374 | sg = asg + 1; | 374 | sg = asg + sglists; |
| 375 | 375 | ||
| 376 | skb->ip_summed = CHECKSUM_NONE; | 376 | skb->ip_summed = CHECKSUM_NONE; |
| 377 | 377 | ||
diff --git a/net/ipv6/route.c b/net/ipv6/route.c index 843406f14d7b..fd0eec6f88c6 100644 --- a/net/ipv6/route.c +++ b/net/ipv6/route.c | |||
| @@ -153,6 +153,12 @@ static void ip6_rt_blackhole_update_pmtu(struct dst_entry *dst, u32 mtu) | |||
| 153 | { | 153 | { |
| 154 | } | 154 | } |
| 155 | 155 | ||
| 156 | static u32 *ip6_rt_blackhole_cow_metrics(struct dst_entry *dst, | ||
| 157 | unsigned long old) | ||
| 158 | { | ||
| 159 | return NULL; | ||
| 160 | } | ||
| 161 | |||
| 156 | static struct dst_ops ip6_dst_blackhole_ops = { | 162 | static struct dst_ops ip6_dst_blackhole_ops = { |
| 157 | .family = AF_INET6, | 163 | .family = AF_INET6, |
| 158 | .protocol = cpu_to_be16(ETH_P_IPV6), | 164 | .protocol = cpu_to_be16(ETH_P_IPV6), |
| @@ -161,6 +167,7 @@ static struct dst_ops ip6_dst_blackhole_ops = { | |||
| 161 | .default_mtu = ip6_blackhole_default_mtu, | 167 | .default_mtu = ip6_blackhole_default_mtu, |
| 162 | .default_advmss = ip6_default_advmss, | 168 | .default_advmss = ip6_default_advmss, |
| 163 | .update_pmtu = ip6_rt_blackhole_update_pmtu, | 169 | .update_pmtu = ip6_rt_blackhole_update_pmtu, |
| 170 | .cow_metrics = ip6_rt_blackhole_cow_metrics, | ||
| 164 | }; | 171 | }; |
| 165 | 172 | ||
| 166 | static const u32 ip6_template_metrics[RTAX_MAX] = { | 173 | static const u32 ip6_template_metrics[RTAX_MAX] = { |
| @@ -2012,7 +2019,6 @@ struct rt6_info *addrconf_dst_alloc(struct inet6_dev *idev, | |||
| 2012 | rt->dst.output = ip6_output; | 2019 | rt->dst.output = ip6_output; |
| 2013 | rt->rt6i_dev = net->loopback_dev; | 2020 | rt->rt6i_dev = net->loopback_dev; |
| 2014 | rt->rt6i_idev = idev; | 2021 | rt->rt6i_idev = idev; |
| 2015 | dst_metric_set(&rt->dst, RTAX_HOPLIMIT, -1); | ||
| 2016 | rt->dst.obsolete = -1; | 2022 | rt->dst.obsolete = -1; |
| 2017 | 2023 | ||
| 2018 | rt->rt6i_flags = RTF_UP | RTF_NONEXTHOP; | 2024 | rt->rt6i_flags = RTF_UP | RTF_NONEXTHOP; |
diff --git a/net/ipv6/udp.c b/net/ipv6/udp.c index 15c37746845e..9e305d74b3d4 100644 --- a/net/ipv6/udp.c +++ b/net/ipv6/udp.c | |||
| @@ -1335,7 +1335,7 @@ static struct sk_buff *udp6_ufo_fragment(struct sk_buff *skb, u32 features) | |||
| 1335 | skb->ip_summed = CHECKSUM_NONE; | 1335 | skb->ip_summed = CHECKSUM_NONE; |
| 1336 | 1336 | ||
| 1337 | /* Check if there is enough headroom to insert fragment header. */ | 1337 | /* Check if there is enough headroom to insert fragment header. */ |
| 1338 | if ((skb_headroom(skb) < frag_hdr_sz) && | 1338 | if ((skb_mac_header(skb) < skb->head + frag_hdr_sz) && |
| 1339 | pskb_expand_head(skb, frag_hdr_sz, 0, GFP_ATOMIC)) | 1339 | pskb_expand_head(skb, frag_hdr_sz, 0, GFP_ATOMIC)) |
| 1340 | goto out; | 1340 | goto out; |
| 1341 | 1341 | ||
diff --git a/net/l2tp/l2tp_ip.c b/net/l2tp/l2tp_ip.c index fce9bd3bd3fe..5c04f3e42704 100644 --- a/net/l2tp/l2tp_ip.c +++ b/net/l2tp/l2tp_ip.c | |||
| @@ -667,7 +667,7 @@ MODULE_AUTHOR("James Chapman <jchapman@katalix.com>"); | |||
| 667 | MODULE_DESCRIPTION("L2TP over IP"); | 667 | MODULE_DESCRIPTION("L2TP over IP"); |
| 668 | MODULE_VERSION("1.0"); | 668 | MODULE_VERSION("1.0"); |
| 669 | 669 | ||
| 670 | /* Use the value of SOCK_DGRAM (2) directory, because __stringify does't like | 670 | /* Use the value of SOCK_DGRAM (2) directory, because __stringify doesn't like |
| 671 | * enums | 671 | * enums |
| 672 | */ | 672 | */ |
| 673 | MODULE_ALIAS_NET_PF_PROTO_TYPE(PF_INET, 2, IPPROTO_L2TP); | 673 | MODULE_ALIAS_NET_PF_PROTO_TYPE(PF_INET, 2, IPPROTO_L2TP); |
diff --git a/net/mac80211/cfg.c b/net/mac80211/cfg.c index 334213571ad0..44049733c4ea 100644 --- a/net/mac80211/cfg.c +++ b/net/mac80211/cfg.c | |||
| @@ -1504,6 +1504,8 @@ int __ieee80211_request_smps(struct ieee80211_sub_if_data *sdata, | |||
| 1504 | enum ieee80211_smps_mode old_req; | 1504 | enum ieee80211_smps_mode old_req; |
| 1505 | int err; | 1505 | int err; |
| 1506 | 1506 | ||
| 1507 | lockdep_assert_held(&sdata->u.mgd.mtx); | ||
| 1508 | |||
| 1507 | old_req = sdata->u.mgd.req_smps; | 1509 | old_req = sdata->u.mgd.req_smps; |
| 1508 | sdata->u.mgd.req_smps = smps_mode; | 1510 | sdata->u.mgd.req_smps = smps_mode; |
| 1509 | 1511 | ||
diff --git a/net/mac80211/debugfs_netdev.c b/net/mac80211/debugfs_netdev.c index dacace6b1393..9ea7c0d0103f 100644 --- a/net/mac80211/debugfs_netdev.c +++ b/net/mac80211/debugfs_netdev.c | |||
| @@ -177,9 +177,9 @@ static int ieee80211_set_smps(struct ieee80211_sub_if_data *sdata, | |||
| 177 | if (sdata->vif.type != NL80211_IFTYPE_STATION) | 177 | if (sdata->vif.type != NL80211_IFTYPE_STATION) |
| 178 | return -EOPNOTSUPP; | 178 | return -EOPNOTSUPP; |
| 179 | 179 | ||
| 180 | mutex_lock(&local->iflist_mtx); | 180 | mutex_lock(&sdata->u.mgd.mtx); |
| 181 | err = __ieee80211_request_smps(sdata, smps_mode); | 181 | err = __ieee80211_request_smps(sdata, smps_mode); |
| 182 | mutex_unlock(&local->iflist_mtx); | 182 | mutex_unlock(&sdata->u.mgd.mtx); |
| 183 | 183 | ||
| 184 | return err; | 184 | return err; |
| 185 | } | 185 | } |
diff --git a/net/sctp/ulpevent.c b/net/sctp/ulpevent.c index dff27d5e22fd..61b1f5ada96a 100644 --- a/net/sctp/ulpevent.c +++ b/net/sctp/ulpevent.c | |||
| @@ -554,7 +554,7 @@ struct sctp_ulpevent *sctp_ulpevent_make_send_failed( | |||
| 554 | memcpy(&ssf->ssf_info, &chunk->sinfo, sizeof(struct sctp_sndrcvinfo)); | 554 | memcpy(&ssf->ssf_info, &chunk->sinfo, sizeof(struct sctp_sndrcvinfo)); |
| 555 | 555 | ||
| 556 | /* Per TSVWG discussion with Randy. Allow the application to | 556 | /* Per TSVWG discussion with Randy. Allow the application to |
| 557 | * resemble a fragmented message. | 557 | * reassemble a fragmented message. |
| 558 | */ | 558 | */ |
| 559 | ssf->ssf_info.sinfo_flags = chunk->chunk_hdr->flags; | 559 | ssf->ssf_info.sinfo_flags = chunk->chunk_hdr->flags; |
| 560 | 560 | ||
diff --git a/net/sunrpc/Kconfig b/net/sunrpc/Kconfig index 8873fd8ddacd..b2198e65d8bb 100644 --- a/net/sunrpc/Kconfig +++ b/net/sunrpc/Kconfig | |||
| @@ -18,14 +18,13 @@ config SUNRPC_XPRT_RDMA | |||
| 18 | If unsure, say N. | 18 | If unsure, say N. |
| 19 | 19 | ||
| 20 | config RPCSEC_GSS_KRB5 | 20 | config RPCSEC_GSS_KRB5 |
| 21 | tristate | 21 | tristate "Secure RPC: Kerberos V mechanism" |
| 22 | depends on SUNRPC && CRYPTO | 22 | depends on SUNRPC && CRYPTO |
| 23 | prompt "Secure RPC: Kerberos V mechanism" if !(NFS_V4 || NFSD_V4) | 23 | depends on CRYPTO_MD5 && CRYPTO_DES && CRYPTO_CBC && CRYPTO_CTS |
| 24 | depends on CRYPTO_ECB && CRYPTO_HMAC && CRYPTO_SHA1 && CRYPTO_AES | ||
| 25 | depends on CRYPTO_ARC4 | ||
| 24 | default y | 26 | default y |
| 25 | select SUNRPC_GSS | 27 | select SUNRPC_GSS |
| 26 | select CRYPTO_MD5 | ||
| 27 | select CRYPTO_DES | ||
| 28 | select CRYPTO_CBC | ||
| 29 | help | 28 | help |
| 30 | Choose Y here to enable Secure RPC using the Kerberos version 5 | 29 | Choose Y here to enable Secure RPC using the Kerberos version 5 |
| 31 | GSS-API mechanism (RFC 1964). | 30 | GSS-API mechanism (RFC 1964). |
diff --git a/net/sunrpc/auth_gss/auth_gss.c b/net/sunrpc/auth_gss/auth_gss.c index f3914d0c5079..339ba64cce1e 100644 --- a/net/sunrpc/auth_gss/auth_gss.c +++ b/net/sunrpc/auth_gss/auth_gss.c | |||
| @@ -520,7 +520,7 @@ gss_refresh_upcall(struct rpc_task *task) | |||
| 520 | warn_gssd(); | 520 | warn_gssd(); |
| 521 | task->tk_timeout = 15*HZ; | 521 | task->tk_timeout = 15*HZ; |
| 522 | rpc_sleep_on(&pipe_version_rpc_waitqueue, task, NULL); | 522 | rpc_sleep_on(&pipe_version_rpc_waitqueue, task, NULL); |
| 523 | return 0; | 523 | return -EAGAIN; |
| 524 | } | 524 | } |
| 525 | if (IS_ERR(gss_msg)) { | 525 | if (IS_ERR(gss_msg)) { |
| 526 | err = PTR_ERR(gss_msg); | 526 | err = PTR_ERR(gss_msg); |
| @@ -563,10 +563,12 @@ retry: | |||
| 563 | if (PTR_ERR(gss_msg) == -EAGAIN) { | 563 | if (PTR_ERR(gss_msg) == -EAGAIN) { |
| 564 | err = wait_event_interruptible_timeout(pipe_version_waitqueue, | 564 | err = wait_event_interruptible_timeout(pipe_version_waitqueue, |
| 565 | pipe_version >= 0, 15*HZ); | 565 | pipe_version >= 0, 15*HZ); |
| 566 | if (pipe_version < 0) { | ||
| 567 | warn_gssd(); | ||
| 568 | err = -EACCES; | ||
| 569 | } | ||
| 566 | if (err) | 570 | if (err) |
| 567 | goto out; | 571 | goto out; |
| 568 | if (pipe_version < 0) | ||
| 569 | warn_gssd(); | ||
| 570 | goto retry; | 572 | goto retry; |
| 571 | } | 573 | } |
| 572 | if (IS_ERR(gss_msg)) { | 574 | if (IS_ERR(gss_msg)) { |
diff --git a/net/sunrpc/clnt.c b/net/sunrpc/clnt.c index e7a96e478f63..8d83f9d48713 100644 --- a/net/sunrpc/clnt.c +++ b/net/sunrpc/clnt.c | |||
| @@ -1508,7 +1508,10 @@ call_timeout(struct rpc_task *task) | |||
| 1508 | if (clnt->cl_chatty) | 1508 | if (clnt->cl_chatty) |
| 1509 | printk(KERN_NOTICE "%s: server %s not responding, timed out\n", | 1509 | printk(KERN_NOTICE "%s: server %s not responding, timed out\n", |
| 1510 | clnt->cl_protname, clnt->cl_server); | 1510 | clnt->cl_protname, clnt->cl_server); |
| 1511 | rpc_exit(task, -EIO); | 1511 | if (task->tk_flags & RPC_TASK_TIMEOUT) |
| 1512 | rpc_exit(task, -ETIMEDOUT); | ||
| 1513 | else | ||
| 1514 | rpc_exit(task, -EIO); | ||
| 1512 | return; | 1515 | return; |
| 1513 | } | 1516 | } |
| 1514 | 1517 | ||
diff --git a/net/sunrpc/xprt.c b/net/sunrpc/xprt.c index 9494c3767356..ce5eb68a9664 100644 --- a/net/sunrpc/xprt.c +++ b/net/sunrpc/xprt.c | |||
| @@ -906,6 +906,7 @@ void xprt_transmit(struct rpc_task *task) | |||
| 906 | } | 906 | } |
| 907 | 907 | ||
| 908 | dprintk("RPC: %5u xmit complete\n", task->tk_pid); | 908 | dprintk("RPC: %5u xmit complete\n", task->tk_pid); |
| 909 | task->tk_flags |= RPC_TASK_SENT; | ||
| 909 | spin_lock_bh(&xprt->transport_lock); | 910 | spin_lock_bh(&xprt->transport_lock); |
| 910 | 911 | ||
| 911 | xprt->ops->set_retrans_timeout(task); | 912 | xprt->ops->set_retrans_timeout(task); |
diff --git a/net/unix/af_unix.c b/net/unix/af_unix.c index 3a43a8304768..b1d75beb7e20 100644 --- a/net/unix/af_unix.c +++ b/net/unix/af_unix.c | |||
| @@ -524,6 +524,8 @@ static int unix_dgram_connect(struct socket *, struct sockaddr *, | |||
| 524 | int, int); | 524 | int, int); |
| 525 | static int unix_seqpacket_sendmsg(struct kiocb *, struct socket *, | 525 | static int unix_seqpacket_sendmsg(struct kiocb *, struct socket *, |
| 526 | struct msghdr *, size_t); | 526 | struct msghdr *, size_t); |
| 527 | static int unix_seqpacket_recvmsg(struct kiocb *, struct socket *, | ||
| 528 | struct msghdr *, size_t, int); | ||
| 527 | 529 | ||
| 528 | static const struct proto_ops unix_stream_ops = { | 530 | static const struct proto_ops unix_stream_ops = { |
| 529 | .family = PF_UNIX, | 531 | .family = PF_UNIX, |
| @@ -583,7 +585,7 @@ static const struct proto_ops unix_seqpacket_ops = { | |||
| 583 | .setsockopt = sock_no_setsockopt, | 585 | .setsockopt = sock_no_setsockopt, |
| 584 | .getsockopt = sock_no_getsockopt, | 586 | .getsockopt = sock_no_getsockopt, |
| 585 | .sendmsg = unix_seqpacket_sendmsg, | 587 | .sendmsg = unix_seqpacket_sendmsg, |
| 586 | .recvmsg = unix_dgram_recvmsg, | 588 | .recvmsg = unix_seqpacket_recvmsg, |
| 587 | .mmap = sock_no_mmap, | 589 | .mmap = sock_no_mmap, |
| 588 | .sendpage = sock_no_sendpage, | 590 | .sendpage = sock_no_sendpage, |
| 589 | }; | 591 | }; |
| @@ -1699,6 +1701,18 @@ static int unix_seqpacket_sendmsg(struct kiocb *kiocb, struct socket *sock, | |||
| 1699 | return unix_dgram_sendmsg(kiocb, sock, msg, len); | 1701 | return unix_dgram_sendmsg(kiocb, sock, msg, len); |
| 1700 | } | 1702 | } |
| 1701 | 1703 | ||
| 1704 | static int unix_seqpacket_recvmsg(struct kiocb *iocb, struct socket *sock, | ||
| 1705 | struct msghdr *msg, size_t size, | ||
| 1706 | int flags) | ||
| 1707 | { | ||
| 1708 | struct sock *sk = sock->sk; | ||
| 1709 | |||
| 1710 | if (sk->sk_state != TCP_ESTABLISHED) | ||
| 1711 | return -ENOTCONN; | ||
| 1712 | |||
| 1713 | return unix_dgram_recvmsg(iocb, sock, msg, size, flags); | ||
| 1714 | } | ||
| 1715 | |||
| 1702 | static void unix_copy_addr(struct msghdr *msg, struct sock *sk) | 1716 | static void unix_copy_addr(struct msghdr *msg, struct sock *sk) |
| 1703 | { | 1717 | { |
| 1704 | struct unix_sock *u = unix_sk(sk); | 1718 | struct unix_sock *u = unix_sk(sk); |
diff --git a/net/xfrm/xfrm_replay.c b/net/xfrm/xfrm_replay.c index f218385950ca..e8a781422feb 100644 --- a/net/xfrm/xfrm_replay.c +++ b/net/xfrm/xfrm_replay.c | |||
| @@ -532,7 +532,7 @@ int xfrm_init_replay(struct xfrm_state *x) | |||
| 532 | 532 | ||
| 533 | if (replay_esn) { | 533 | if (replay_esn) { |
| 534 | if (replay_esn->replay_window > | 534 | if (replay_esn->replay_window > |
| 535 | replay_esn->bmp_len * sizeof(__u32)) | 535 | replay_esn->bmp_len * sizeof(__u32) * 8) |
| 536 | return -EINVAL; | 536 | return -EINVAL; |
| 537 | 537 | ||
| 538 | if ((x->props.flags & XFRM_STATE_ESN) && x->replay_esn) | 538 | if ((x->props.flags & XFRM_STATE_ESN) && x->replay_esn) |
diff --git a/net/xfrm/xfrm_user.c b/net/xfrm/xfrm_user.c index 5d1d60d3ca83..c658cb3bc7c3 100644 --- a/net/xfrm/xfrm_user.c +++ b/net/xfrm/xfrm_user.c | |||
| @@ -124,6 +124,9 @@ static inline int verify_replay(struct xfrm_usersa_info *p, | |||
| 124 | { | 124 | { |
| 125 | struct nlattr *rt = attrs[XFRMA_REPLAY_ESN_VAL]; | 125 | struct nlattr *rt = attrs[XFRMA_REPLAY_ESN_VAL]; |
| 126 | 126 | ||
| 127 | if ((p->flags & XFRM_STATE_ESN) && !rt) | ||
| 128 | return -EINVAL; | ||
| 129 | |||
| 127 | if (!rt) | 130 | if (!rt) |
| 128 | return 0; | 131 | return 0; |
| 129 | 132 | ||
diff --git a/sound/aoa/codecs/tas.c b/sound/aoa/codecs/tas.c index 58804c7acfcf..fd2188c3df2b 100644 --- a/sound/aoa/codecs/tas.c +++ b/sound/aoa/codecs/tas.c | |||
| @@ -170,7 +170,7 @@ static void tas_set_volume(struct tas *tas) | |||
| 170 | /* analysing the volume and mixer tables shows | 170 | /* analysing the volume and mixer tables shows |
| 171 | * that they are similar enough when we shift | 171 | * that they are similar enough when we shift |
| 172 | * the mixer table down by 4 bits. The error | 172 | * the mixer table down by 4 bits. The error |
| 173 | * is minuscule, in just one item the error | 173 | * is miniscule, in just one item the error |
| 174 | * is 1, at a value of 0x07f17b (mixer table | 174 | * is 1, at a value of 0x07f17b (mixer table |
| 175 | * value is 0x07f17a) */ | 175 | * value is 0x07f17a) */ |
| 176 | tmp = tas_gaintable[left]; | 176 | tmp = tas_gaintable[left]; |
diff --git a/sound/pci/au88x0/au88x0_pcm.c b/sound/pci/au88x0/au88x0_pcm.c index 33f0ba5559a7..62e959120c44 100644 --- a/sound/pci/au88x0/au88x0_pcm.c +++ b/sound/pci/au88x0/au88x0_pcm.c | |||
| @@ -44,10 +44,10 @@ static struct snd_pcm_hardware snd_vortex_playback_hw_adb = { | |||
| 44 | .channels_min = 1, | 44 | .channels_min = 1, |
| 45 | .channels_max = 2, | 45 | .channels_max = 2, |
| 46 | .buffer_bytes_max = 0x10000, | 46 | .buffer_bytes_max = 0x10000, |
| 47 | .period_bytes_min = 0x1, | 47 | .period_bytes_min = 0x20, |
| 48 | .period_bytes_max = 0x1000, | 48 | .period_bytes_max = 0x1000, |
| 49 | .periods_min = 2, | 49 | .periods_min = 2, |
| 50 | .periods_max = 32, | 50 | .periods_max = 1024, |
| 51 | }; | 51 | }; |
| 52 | 52 | ||
| 53 | #ifndef CHIP_AU8820 | 53 | #ifndef CHIP_AU8820 |
| @@ -140,6 +140,9 @@ static int snd_vortex_pcm_open(struct snd_pcm_substream *substream) | |||
| 140 | SNDRV_PCM_HW_PARAM_PERIOD_BYTES)) < 0) | 140 | SNDRV_PCM_HW_PARAM_PERIOD_BYTES)) < 0) |
| 141 | return err; | 141 | return err; |
| 142 | 142 | ||
| 143 | snd_pcm_hw_constraint_step(runtime, 0, | ||
| 144 | SNDRV_PCM_HW_PARAM_BUFFER_BYTES, 64); | ||
| 145 | |||
| 143 | if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) { | 146 | if (VORTEX_PCM_TYPE(substream->pcm) != VORTEX_PCM_WT) { |
| 144 | #ifndef CHIP_AU8820 | 147 | #ifndef CHIP_AU8820 |
| 145 | if (VORTEX_PCM_TYPE(substream->pcm) == VORTEX_PCM_A3D) { | 148 | if (VORTEX_PCM_TYPE(substream->pcm) == VORTEX_PCM_A3D) { |
diff --git a/sound/pci/hda/patch_realtek.c b/sound/pci/hda/patch_realtek.c index d3bd2c10180f..c82979a8cd09 100644 --- a/sound/pci/hda/patch_realtek.c +++ b/sound/pci/hda/patch_realtek.c | |||
| @@ -1704,11 +1704,11 @@ static void alc_apply_fixup(struct hda_codec *codec, int action) | |||
| 1704 | codec->chip_name, fix->type); | 1704 | codec->chip_name, fix->type); |
| 1705 | break; | 1705 | break; |
| 1706 | } | 1706 | } |
| 1707 | if (!fix[id].chained) | 1707 | if (!fix->chained) |
| 1708 | break; | 1708 | break; |
| 1709 | if (++depth > 10) | 1709 | if (++depth > 10) |
| 1710 | break; | 1710 | break; |
| 1711 | id = fix[id].chain_id; | 1711 | id = fix->chain_id; |
| 1712 | } | 1712 | } |
| 1713 | } | 1713 | } |
| 1714 | 1714 | ||
| @@ -5645,6 +5645,7 @@ static void fillup_priv_adc_nids(struct hda_codec *codec, hda_nid_t *nids, | |||
| 5645 | static struct snd_pci_quirk beep_white_list[] = { | 5645 | static struct snd_pci_quirk beep_white_list[] = { |
| 5646 | SND_PCI_QUIRK(0x1043, 0x829f, "ASUS", 1), | 5646 | SND_PCI_QUIRK(0x1043, 0x829f, "ASUS", 1), |
| 5647 | SND_PCI_QUIRK(0x1043, 0x83ce, "EeePC", 1), | 5647 | SND_PCI_QUIRK(0x1043, 0x83ce, "EeePC", 1), |
| 5648 | SND_PCI_QUIRK(0x1043, 0x831a, "EeePC", 1), | ||
| 5648 | SND_PCI_QUIRK(0x8086, 0xd613, "Intel", 1), | 5649 | SND_PCI_QUIRK(0x8086, 0xd613, "Intel", 1), |
| 5649 | {} | 5650 | {} |
| 5650 | }; | 5651 | }; |
| @@ -9863,6 +9864,7 @@ static struct snd_pci_quirk alc882_cfg_tbl[] = { | |||
| 9863 | SND_PCI_QUIRK(0x1071, 0x8258, "Evesham Voyaeger", ALC883_LAPTOP_EAPD), | 9864 | SND_PCI_QUIRK(0x1071, 0x8258, "Evesham Voyaeger", ALC883_LAPTOP_EAPD), |
| 9864 | SND_PCI_QUIRK(0x10f1, 0x2350, "TYAN-S2350", ALC888_6ST_DELL), | 9865 | SND_PCI_QUIRK(0x10f1, 0x2350, "TYAN-S2350", ALC888_6ST_DELL), |
| 9865 | SND_PCI_QUIRK(0x108e, 0x534d, NULL, ALC883_3ST_6ch), | 9866 | SND_PCI_QUIRK(0x108e, 0x534d, NULL, ALC883_3ST_6ch), |
| 9867 | SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte P35 DS3R", ALC882_6ST_DIG), | ||
| 9866 | 9868 | ||
| 9867 | SND_PCI_QUIRK(0x1462, 0x0349, "MSI", ALC883_TARGA_2ch_DIG), | 9869 | SND_PCI_QUIRK(0x1462, 0x0349, "MSI", ALC883_TARGA_2ch_DIG), |
| 9868 | SND_PCI_QUIRK(0x1462, 0x040d, "MSI", ALC883_TARGA_2ch_DIG), | 9870 | SND_PCI_QUIRK(0x1462, 0x040d, "MSI", ALC883_TARGA_2ch_DIG), |
| @@ -10699,7 +10701,6 @@ enum { | |||
| 10699 | PINFIX_LENOVO_Y530, | 10701 | PINFIX_LENOVO_Y530, |
| 10700 | PINFIX_PB_M5210, | 10702 | PINFIX_PB_M5210, |
| 10701 | PINFIX_ACER_ASPIRE_7736, | 10703 | PINFIX_ACER_ASPIRE_7736, |
| 10702 | PINFIX_GIGABYTE_880GM, | ||
| 10703 | }; | 10704 | }; |
| 10704 | 10705 | ||
| 10705 | static const struct alc_fixup alc882_fixups[] = { | 10706 | static const struct alc_fixup alc882_fixups[] = { |
| @@ -10731,13 +10732,6 @@ static const struct alc_fixup alc882_fixups[] = { | |||
| 10731 | .type = ALC_FIXUP_SKU, | 10732 | .type = ALC_FIXUP_SKU, |
| 10732 | .v.sku = ALC_FIXUP_SKU_IGNORE, | 10733 | .v.sku = ALC_FIXUP_SKU_IGNORE, |
| 10733 | }, | 10734 | }, |
| 10734 | [PINFIX_GIGABYTE_880GM] = { | ||
| 10735 | .type = ALC_FIXUP_PINS, | ||
| 10736 | .v.pins = (const struct alc_pincfg[]) { | ||
| 10737 | { 0x14, 0x1114410 }, /* set as speaker */ | ||
| 10738 | { } | ||
| 10739 | } | ||
| 10740 | }, | ||
| 10741 | }; | 10735 | }; |
| 10742 | 10736 | ||
| 10743 | static struct snd_pci_quirk alc882_fixup_tbl[] = { | 10737 | static struct snd_pci_quirk alc882_fixup_tbl[] = { |
| @@ -10745,7 +10739,6 @@ static struct snd_pci_quirk alc882_fixup_tbl[] = { | |||
| 10745 | SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Y530", PINFIX_LENOVO_Y530), | 10739 | SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Y530", PINFIX_LENOVO_Y530), |
| 10746 | SND_PCI_QUIRK(0x147b, 0x107a, "Abit AW9D-MAX", PINFIX_ABIT_AW9D_MAX), | 10740 | SND_PCI_QUIRK(0x147b, 0x107a, "Abit AW9D-MAX", PINFIX_ABIT_AW9D_MAX), |
| 10747 | SND_PCI_QUIRK(0x1025, 0x0296, "Acer Aspire 7736z", PINFIX_ACER_ASPIRE_7736), | 10741 | SND_PCI_QUIRK(0x1025, 0x0296, "Acer Aspire 7736z", PINFIX_ACER_ASPIRE_7736), |
| 10748 | SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte", PINFIX_GIGABYTE_880GM), | ||
| 10749 | {} | 10742 | {} |
| 10750 | }; | 10743 | }; |
| 10751 | 10744 | ||
| @@ -18805,6 +18798,8 @@ static struct snd_pci_quirk alc662_cfg_tbl[] = { | |||
| 18805 | ALC662_3ST_6ch_DIG), | 18798 | ALC662_3ST_6ch_DIG), |
| 18806 | SND_PCI_QUIRK(0x1179, 0xff6e, "Toshiba NB20x", ALC662_AUTO), | 18799 | SND_PCI_QUIRK(0x1179, 0xff6e, "Toshiba NB20x", ALC662_AUTO), |
| 18807 | SND_PCI_QUIRK(0x144d, 0xca00, "Samsung NC10", ALC272_SAMSUNG_NC10), | 18800 | SND_PCI_QUIRK(0x144d, 0xca00, "Samsung NC10", ALC272_SAMSUNG_NC10), |
| 18801 | SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte 945GCM-S2L", | ||
| 18802 | ALC662_3ST_6ch_DIG), | ||
| 18808 | SND_PCI_QUIRK(0x152d, 0x2304, "Quanta WH1", ALC663_ASUS_H13), | 18803 | SND_PCI_QUIRK(0x152d, 0x2304, "Quanta WH1", ALC663_ASUS_H13), |
| 18809 | SND_PCI_QUIRK(0x1565, 0x820f, "Biostar TA780G M2+", ALC662_3ST_6ch_DIG), | 18804 | SND_PCI_QUIRK(0x1565, 0x820f, "Biostar TA780G M2+", ALC662_3ST_6ch_DIG), |
| 18810 | SND_PCI_QUIRK(0x1631, 0xc10c, "PB RS65", ALC663_ASUS_M51VA), | 18805 | SND_PCI_QUIRK(0x1631, 0xc10c, "PB RS65", ALC663_ASUS_M51VA), |
| @@ -19478,7 +19473,7 @@ enum { | |||
| 19478 | ALC662_FIXUP_IDEAPAD, | 19473 | ALC662_FIXUP_IDEAPAD, |
| 19479 | ALC272_FIXUP_MARIO, | 19474 | ALC272_FIXUP_MARIO, |
| 19480 | ALC662_FIXUP_CZC_P10T, | 19475 | ALC662_FIXUP_CZC_P10T, |
| 19481 | ALC662_FIXUP_GIGABYTE, | 19476 | ALC662_FIXUP_SKU_IGNORE, |
| 19482 | }; | 19477 | }; |
| 19483 | 19478 | ||
| 19484 | static const struct alc_fixup alc662_fixups[] = { | 19479 | static const struct alc_fixup alc662_fixups[] = { |
| @@ -19507,20 +19502,17 @@ static const struct alc_fixup alc662_fixups[] = { | |||
| 19507 | {} | 19502 | {} |
| 19508 | } | 19503 | } |
| 19509 | }, | 19504 | }, |
| 19510 | [ALC662_FIXUP_GIGABYTE] = { | 19505 | [ALC662_FIXUP_SKU_IGNORE] = { |
| 19511 | .type = ALC_FIXUP_PINS, | 19506 | .type = ALC_FIXUP_SKU, |
| 19512 | .v.pins = (const struct alc_pincfg[]) { | 19507 | .v.sku = ALC_FIXUP_SKU_IGNORE, |
| 19513 | { 0x14, 0x1114410 }, /* set as speaker */ | ||
| 19514 | { } | ||
| 19515 | } | ||
| 19516 | }, | 19508 | }, |
| 19517 | }; | 19509 | }; |
| 19518 | 19510 | ||
| 19519 | static struct snd_pci_quirk alc662_fixup_tbl[] = { | 19511 | static struct snd_pci_quirk alc662_fixup_tbl[] = { |
| 19520 | SND_PCI_QUIRK(0x1025, 0x0308, "Acer Aspire 8942G", ALC662_FIXUP_ASPIRE), | 19512 | SND_PCI_QUIRK(0x1025, 0x0308, "Acer Aspire 8942G", ALC662_FIXUP_ASPIRE), |
| 19513 | SND_PCI_QUIRK(0x1025, 0x031c, "Gateway NV79", ALC662_FIXUP_SKU_IGNORE), | ||
| 19521 | SND_PCI_QUIRK(0x1025, 0x038b, "Acer Aspire 8943G", ALC662_FIXUP_ASPIRE), | 19514 | SND_PCI_QUIRK(0x1025, 0x038b, "Acer Aspire 8943G", ALC662_FIXUP_ASPIRE), |
| 19522 | SND_PCI_QUIRK(0x144d, 0xc051, "Samsung R720", ALC662_FIXUP_IDEAPAD), | 19515 | SND_PCI_QUIRK(0x144d, 0xc051, "Samsung R720", ALC662_FIXUP_IDEAPAD), |
| 19523 | SND_PCI_QUIRK(0x1458, 0xa002, "Gigabyte", ALC662_FIXUP_GIGABYTE), | ||
| 19524 | SND_PCI_QUIRK(0x17aa, 0x38af, "Lenovo Ideapad Y550P", ALC662_FIXUP_IDEAPAD), | 19516 | SND_PCI_QUIRK(0x17aa, 0x38af, "Lenovo Ideapad Y550P", ALC662_FIXUP_IDEAPAD), |
| 19525 | SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Ideapad Y550", ALC662_FIXUP_IDEAPAD), | 19517 | SND_PCI_QUIRK(0x17aa, 0x3a0d, "Lenovo Ideapad Y550", ALC662_FIXUP_IDEAPAD), |
| 19526 | SND_PCI_QUIRK(0x1b35, 0x2206, "CZC P10T", ALC662_FIXUP_CZC_P10T), | 19518 | SND_PCI_QUIRK(0x1b35, 0x2206, "CZC P10T", ALC662_FIXUP_CZC_P10T), |
diff --git a/sound/pci/hda/patch_via.c b/sound/pci/hda/patch_via.c index 1371b57c11e8..0997031c48d2 100644 --- a/sound/pci/hda/patch_via.c +++ b/sound/pci/hda/patch_via.c | |||
| @@ -1292,14 +1292,18 @@ static void notify_aa_path_ctls(struct hda_codec *codec) | |||
| 1292 | { | 1292 | { |
| 1293 | int i; | 1293 | int i; |
| 1294 | struct snd_ctl_elem_id id; | 1294 | struct snd_ctl_elem_id id; |
| 1295 | const char *labels[] = {"Mic", "Front Mic", "Line"}; | 1295 | const char *labels[] = {"Mic", "Front Mic", "Line", "Rear Mic"}; |
| 1296 | struct snd_kcontrol *ctl; | ||
| 1296 | 1297 | ||
| 1297 | memset(&id, 0, sizeof(id)); | 1298 | memset(&id, 0, sizeof(id)); |
| 1298 | id.iface = SNDRV_CTL_ELEM_IFACE_MIXER; | 1299 | id.iface = SNDRV_CTL_ELEM_IFACE_MIXER; |
| 1299 | for (i = 0; i < ARRAY_SIZE(labels); i++) { | 1300 | for (i = 0; i < ARRAY_SIZE(labels); i++) { |
| 1300 | sprintf(id.name, "%s Playback Volume", labels[i]); | 1301 | sprintf(id.name, "%s Playback Volume", labels[i]); |
| 1301 | snd_ctl_notify(codec->bus->card, SNDRV_CTL_EVENT_MASK_VALUE, | 1302 | ctl = snd_hda_find_mixer_ctl(codec, id.name); |
| 1302 | &id); | 1303 | if (ctl) |
| 1304 | snd_ctl_notify(codec->bus->card, | ||
| 1305 | SNDRV_CTL_EVENT_MASK_VALUE, | ||
| 1306 | &ctl->id); | ||
| 1303 | } | 1307 | } |
| 1304 | } | 1308 | } |
| 1305 | 1309 | ||
diff --git a/sound/usb/format.c b/sound/usb/format.c index 5b792d2c8061..f079b5e2ab28 100644 --- a/sound/usb/format.c +++ b/sound/usb/format.c | |||
| @@ -176,9 +176,11 @@ static int parse_audio_format_rates_v1(struct snd_usb_audio *chip, struct audiof | |||
| 176 | if (!rate) | 176 | if (!rate) |
| 177 | continue; | 177 | continue; |
| 178 | /* C-Media CM6501 mislabels its 96 kHz altsetting */ | 178 | /* C-Media CM6501 mislabels its 96 kHz altsetting */ |
| 179 | /* Terratec Aureon 7.1 USB C-Media 6206, too */ | ||
| 179 | if (rate == 48000 && nr_rates == 1 && | 180 | if (rate == 48000 && nr_rates == 1 && |
| 180 | (chip->usb_id == USB_ID(0x0d8c, 0x0201) || | 181 | (chip->usb_id == USB_ID(0x0d8c, 0x0201) || |
| 181 | chip->usb_id == USB_ID(0x0d8c, 0x0102)) && | 182 | chip->usb_id == USB_ID(0x0d8c, 0x0102) || |
| 183 | chip->usb_id == USB_ID(0x0ccd, 0x00b1)) && | ||
| 182 | fp->altsetting == 5 && fp->maxpacksize == 392) | 184 | fp->altsetting == 5 && fp->maxpacksize == 392) |
| 183 | rate = 96000; | 185 | rate = 96000; |
| 184 | /* Creative VF0470 Live Cam reports 16 kHz instead of 8kHz */ | 186 | /* Creative VF0470 Live Cam reports 16 kHz instead of 8kHz */ |
diff --git a/sound/usb/quirks.c b/sound/usb/quirks.c index ec07e62e53f3..1b94ec3a3368 100644 --- a/sound/usb/quirks.c +++ b/sound/usb/quirks.c | |||
| @@ -533,6 +533,7 @@ int snd_usb_apply_boot_quirk(struct usb_device *dev, | |||
| 533 | 533 | ||
| 534 | case USB_ID(0x0d8c, 0x0102): | 534 | case USB_ID(0x0d8c, 0x0102): |
| 535 | /* C-Media CM6206 / CM106-Like Sound Device */ | 535 | /* C-Media CM6206 / CM106-Like Sound Device */ |
| 536 | case USB_ID(0x0ccd, 0x00b1): /* Terratec Aureon 7.1 USB */ | ||
| 536 | return snd_usb_cm6206_boot_quirk(dev); | 537 | return snd_usb_cm6206_boot_quirk(dev); |
| 537 | 538 | ||
| 538 | case USB_ID(0x133e, 0x0815): | 539 | case USB_ID(0x133e, 0x0815): |
