blob: e3c8b05dcae482e3148dc5643a82cc8d78d4888a [file] [log] [blame]
Ben Skeggs147dc382010-12-16 13:32:36 +10001/*
2 * Copyright 2010 Red Hat Inc.
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
13 *
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
21 *
22 * Authors: Ben Skeggs
23 */
24
Ben Skeggs6ee73862009-12-11 19:24:15 +100025#include "drmP.h"
26#include "nouveau_drv.h"
27#include "nouveau_dma.h"
Ben Skeggsa8eaebc2010-09-01 15:24:31 +100028#include "nouveau_ramht.h"
Ben Skeggs6ee73862009-12-11 19:24:15 +100029#include "nouveau_fbcon.h"
Ben Skeggs4c1361422010-11-15 11:54:21 +100030#include "nouveau_mm.h"
Ben Skeggs6ee73862009-12-11 19:24:15 +100031
Ben Skeggsceed5f32010-10-05 16:41:29 +100032int
Ben Skeggs6ee73862009-12-11 19:24:15 +100033nv50_fbcon_fillrect(struct fb_info *info, const struct fb_fillrect *rect)
34{
Dave Airlie8be48d92010-03-30 05:34:14 +000035 struct nouveau_fbdev *nfbdev = info->par;
36 struct drm_device *dev = nfbdev->dev;
Ben Skeggs6ee73862009-12-11 19:24:15 +100037 struct drm_nouveau_private *dev_priv = dev->dev_private;
38 struct nouveau_channel *chan = dev_priv->channel;
Ben Skeggsceed5f32010-10-05 16:41:29 +100039 int ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +100040
Ben Skeggsceed5f32010-10-05 16:41:29 +100041 ret = RING_SPACE(chan, rect->rop == ROP_COPY ? 7 : 11);
42 if (ret)
43 return ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +100044
45 if (rect->rop != ROP_COPY) {
Ben Skeggs6d597022012-04-01 21:09:13 +100046 BEGIN_NV04(chan, NvSub2D, 0x02ac, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +100047 OUT_RING(chan, 1);
48 }
Ben Skeggs6d597022012-04-01 21:09:13 +100049 BEGIN_NV04(chan, NvSub2D, 0x0588, 1);
Ben Skeggsbf5302b2010-01-04 09:10:55 +100050 if (info->fix.visual == FB_VISUAL_TRUECOLOR ||
51 info->fix.visual == FB_VISUAL_DIRECTCOLOR)
52 OUT_RING(chan, ((uint32_t *)info->pseudo_palette)[rect->color]);
53 else
54 OUT_RING(chan, rect->color);
Ben Skeggs6d597022012-04-01 21:09:13 +100055 BEGIN_NV04(chan, NvSub2D, 0x0600, 4);
Ben Skeggs6ee73862009-12-11 19:24:15 +100056 OUT_RING(chan, rect->dx);
57 OUT_RING(chan, rect->dy);
58 OUT_RING(chan, rect->dx + rect->width);
59 OUT_RING(chan, rect->dy + rect->height);
60 if (rect->rop != ROP_COPY) {
Ben Skeggs6d597022012-04-01 21:09:13 +100061 BEGIN_NV04(chan, NvSub2D, 0x02ac, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +100062 OUT_RING(chan, 3);
63 }
64 FIRE_RING(chan);
Ben Skeggsceed5f32010-10-05 16:41:29 +100065 return 0;
Ben Skeggs6ee73862009-12-11 19:24:15 +100066}
67
Ben Skeggsceed5f32010-10-05 16:41:29 +100068int
Ben Skeggs6ee73862009-12-11 19:24:15 +100069nv50_fbcon_copyarea(struct fb_info *info, const struct fb_copyarea *region)
70{
Dave Airlie8be48d92010-03-30 05:34:14 +000071 struct nouveau_fbdev *nfbdev = info->par;
72 struct drm_device *dev = nfbdev->dev;
Ben Skeggs6ee73862009-12-11 19:24:15 +100073 struct drm_nouveau_private *dev_priv = dev->dev_private;
74 struct nouveau_channel *chan = dev_priv->channel;
Ben Skeggsceed5f32010-10-05 16:41:29 +100075 int ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +100076
Ben Skeggsceed5f32010-10-05 16:41:29 +100077 ret = RING_SPACE(chan, 12);
78 if (ret)
79 return ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +100080
Ben Skeggs6d597022012-04-01 21:09:13 +100081 BEGIN_NV04(chan, NvSub2D, 0x0110, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +100082 OUT_RING(chan, 0);
Ben Skeggs6d597022012-04-01 21:09:13 +100083 BEGIN_NV04(chan, NvSub2D, 0x08b0, 4);
Ben Skeggs6ee73862009-12-11 19:24:15 +100084 OUT_RING(chan, region->dx);
85 OUT_RING(chan, region->dy);
86 OUT_RING(chan, region->width);
87 OUT_RING(chan, region->height);
Ben Skeggs6d597022012-04-01 21:09:13 +100088 BEGIN_NV04(chan, NvSub2D, 0x08d0, 4);
Ben Skeggs6ee73862009-12-11 19:24:15 +100089 OUT_RING(chan, 0);
90 OUT_RING(chan, region->sx);
91 OUT_RING(chan, 0);
92 OUT_RING(chan, region->sy);
93 FIRE_RING(chan);
Ben Skeggsceed5f32010-10-05 16:41:29 +100094 return 0;
Ben Skeggs6ee73862009-12-11 19:24:15 +100095}
96
Ben Skeggsceed5f32010-10-05 16:41:29 +100097int
Ben Skeggs6ee73862009-12-11 19:24:15 +100098nv50_fbcon_imageblit(struct fb_info *info, const struct fb_image *image)
99{
Dave Airlie8be48d92010-03-30 05:34:14 +0000100 struct nouveau_fbdev *nfbdev = info->par;
101 struct drm_device *dev = nfbdev->dev;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000102 struct drm_nouveau_private *dev_priv = dev->dev_private;
103 struct nouveau_channel *chan = dev_priv->channel;
104 uint32_t width, dwords, *data = (uint32_t *)image->data;
105 uint32_t mask = ~(~0 >> (32 - info->var.bits_per_pixel));
106 uint32_t *palette = info->pseudo_palette;
Ben Skeggsceed5f32010-10-05 16:41:29 +1000107 int ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000108
Ben Skeggsceed5f32010-10-05 16:41:29 +1000109 if (image->depth != 1)
110 return -ENODEV;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000111
Ben Skeggsceed5f32010-10-05 16:41:29 +1000112 ret = RING_SPACE(chan, 11);
113 if (ret)
114 return ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000115
Matt Turner3bfc7d22010-02-24 23:27:10 -0500116 width = ALIGN(image->width, 32);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000117 dwords = (width * image->height) >> 5;
118
Ben Skeggs6d597022012-04-01 21:09:13 +1000119 BEGIN_NV04(chan, NvSub2D, 0x0814, 2);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000120 if (info->fix.visual == FB_VISUAL_TRUECOLOR ||
121 info->fix.visual == FB_VISUAL_DIRECTCOLOR) {
122 OUT_RING(chan, palette[image->bg_color] | mask);
123 OUT_RING(chan, palette[image->fg_color] | mask);
124 } else {
125 OUT_RING(chan, image->bg_color);
126 OUT_RING(chan, image->fg_color);
127 }
Ben Skeggs6d597022012-04-01 21:09:13 +1000128 BEGIN_NV04(chan, NvSub2D, 0x0838, 2);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000129 OUT_RING(chan, image->width);
130 OUT_RING(chan, image->height);
Ben Skeggs6d597022012-04-01 21:09:13 +1000131 BEGIN_NV04(chan, NvSub2D, 0x0850, 4);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000132 OUT_RING(chan, 0);
133 OUT_RING(chan, image->dx);
134 OUT_RING(chan, 0);
135 OUT_RING(chan, image->dy);
136
137 while (dwords) {
138 int push = dwords > 2047 ? 2047 : dwords;
139
Ben Skeggsceed5f32010-10-05 16:41:29 +1000140 ret = RING_SPACE(chan, push + 1);
141 if (ret)
142 return ret;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000143
144 dwords -= push;
145
Ben Skeggs6d597022012-04-01 21:09:13 +1000146 BEGIN_NI04(chan, NvSub2D, 0x0860, push);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000147 OUT_RINGp(chan, data, push);
148 data += push;
149 }
150
151 FIRE_RING(chan);
Ben Skeggsceed5f32010-10-05 16:41:29 +1000152 return 0;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000153}
154
155int
156nv50_fbcon_accel_init(struct fb_info *info)
157{
Dave Airlie8be48d92010-03-30 05:34:14 +0000158 struct nouveau_fbdev *nfbdev = info->par;
159 struct drm_device *dev = nfbdev->dev;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000160 struct drm_nouveau_private *dev_priv = dev->dev_private;
161 struct nouveau_channel *chan = dev_priv->channel;
Ben Skeggs45143cb2011-06-07 13:12:44 +1000162 struct nouveau_framebuffer *fb = &nfbdev->nouveau_fb;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000163 int ret, format;
Ben Skeggs0c324972010-03-16 13:20:58 +1000164
Ben Skeggs6ee73862009-12-11 19:24:15 +1000165 switch (info->var.bits_per_pixel) {
166 case 8:
167 format = 0xf3;
168 break;
169 case 15:
170 format = 0xf8;
171 break;
172 case 16:
173 format = 0xe8;
174 break;
175 case 32:
176 switch (info->var.transp.length) {
177 case 0: /* depth 24 */
178 case 8: /* depth 32, just use 24.. */
179 format = 0xe6;
180 break;
181 case 2: /* depth 30 */
182 format = 0xd1;
183 break;
184 default:
185 return -EINVAL;
186 }
187 break;
188 default:
189 return -EINVAL;
190 }
191
Ben Skeggsceac3092010-11-23 10:10:24 +1000192 ret = nouveau_gpuobj_gr_new(dev_priv->channel, Nv2D, 0x502d);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000193 if (ret)
194 return ret;
195
196 ret = RING_SPACE(chan, 59);
197 if (ret) {
Marcin Slusarz846975a2010-01-04 19:25:09 +0100198 nouveau_fbcon_gpu_lockup(info);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000199 return ret;
200 }
201
Ben Skeggs6d597022012-04-01 21:09:13 +1000202 BEGIN_NV04(chan, NvSub2D, 0x0000, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000203 OUT_RING(chan, Nv2D);
Ben Skeggsd1b167e2012-05-04 14:01:52 +1000204 BEGIN_NV04(chan, NvSub2D, 0x0184, 3);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000205 OUT_RING(chan, chan->vram_handle);
206 OUT_RING(chan, chan->vram_handle);
207 OUT_RING(chan, chan->vram_handle);
Ben Skeggs6d597022012-04-01 21:09:13 +1000208 BEGIN_NV04(chan, NvSub2D, 0x0290, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000209 OUT_RING(chan, 0);
Ben Skeggs6d597022012-04-01 21:09:13 +1000210 BEGIN_NV04(chan, NvSub2D, 0x0888, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000211 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000212 BEGIN_NV04(chan, NvSub2D, 0x02ac, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000213 OUT_RING(chan, 3);
Ben Skeggs6d597022012-04-01 21:09:13 +1000214 BEGIN_NV04(chan, NvSub2D, 0x02a0, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000215 OUT_RING(chan, 0x55);
Ben Skeggs6d597022012-04-01 21:09:13 +1000216 BEGIN_NV04(chan, NvSub2D, 0x08c0, 4);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000217 OUT_RING(chan, 0);
218 OUT_RING(chan, 1);
219 OUT_RING(chan, 0);
220 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000221 BEGIN_NV04(chan, NvSub2D, 0x0580, 2);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000222 OUT_RING(chan, 4);
223 OUT_RING(chan, format);
Ben Skeggs6d597022012-04-01 21:09:13 +1000224 BEGIN_NV04(chan, NvSub2D, 0x02e8, 2);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000225 OUT_RING(chan, 2);
226 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000227 BEGIN_NV04(chan, NvSub2D, 0x0804, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000228 OUT_RING(chan, format);
Ben Skeggs6d597022012-04-01 21:09:13 +1000229 BEGIN_NV04(chan, NvSub2D, 0x0800, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000230 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000231 BEGIN_NV04(chan, NvSub2D, 0x0808, 3);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000232 OUT_RING(chan, 0);
233 OUT_RING(chan, 0);
Marcin Koƛcielnickic82b88d2010-02-27 18:13:35 +0000234 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000235 BEGIN_NV04(chan, NvSub2D, 0x081c, 1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000236 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000237 BEGIN_NV04(chan, NvSub2D, 0x0840, 4);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000238 OUT_RING(chan, 0);
239 OUT_RING(chan, 1);
240 OUT_RING(chan, 0);
241 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000242 BEGIN_NV04(chan, NvSub2D, 0x0200, 2);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000243 OUT_RING(chan, format);
244 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000245 BEGIN_NV04(chan, NvSub2D, 0x0214, 5);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000246 OUT_RING(chan, info->fix.line_length);
247 OUT_RING(chan, info->var.xres_virtual);
248 OUT_RING(chan, info->var.yres_virtual);
Ben Skeggs45143cb2011-06-07 13:12:44 +1000249 OUT_RING(chan, upper_32_bits(fb->vma.offset));
250 OUT_RING(chan, lower_32_bits(fb->vma.offset));
Ben Skeggs6d597022012-04-01 21:09:13 +1000251 BEGIN_NV04(chan, NvSub2D, 0x0230, 2);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000252 OUT_RING(chan, format);
253 OUT_RING(chan, 1);
Ben Skeggs6d597022012-04-01 21:09:13 +1000254 BEGIN_NV04(chan, NvSub2D, 0x0244, 5);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000255 OUT_RING(chan, info->fix.line_length);
256 OUT_RING(chan, info->var.xres_virtual);
257 OUT_RING(chan, info->var.yres_virtual);
Ben Skeggs45143cb2011-06-07 13:12:44 +1000258 OUT_RING(chan, upper_32_bits(fb->vma.offset));
259 OUT_RING(chan, lower_32_bits(fb->vma.offset));
Ben Skeggs6ee73862009-12-11 19:24:15 +1000260
Ben Skeggs6ee73862009-12-11 19:24:15 +1000261 return 0;
262}
263