diff options
author | Ben Skeggs <bskeggs@redhat.com> | 2014-09-15 15:15:09 +1000 |
---|---|---|
committer | Ben Skeggs <bskeggs@redhat.com> | 2014-10-02 13:25:45 +1000 |
commit | 92b6a983ac04ce945dc2f1556cc77a76323ab4c4 (patch) | |
tree | 79a443e18ba075194da7d758e28d7edcfd5ce148 | |
parent | d8279dfb5b9417a9a5cc7e0b61eaf2cf3f9de5e3 (diff) | |
download | nouveau-92b6a983ac04ce945dc2f1556cc77a76323ab4c4.tar.gz |
disp/gk104-: infoframe registers moved yet again on kepler
Thanks to Vincent Pelletier for pointing this out and providing a proof of
concept patch on the list.
Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
-rw-r--r-- | drm/Kbuild | 1 | ||||
l--------- | drm/core/engine/disp/hdminve0.c | 1 | ||||
-rw-r--r-- | nvkm/engine/disp/Makefile.am | 1 | ||||
-rw-r--r-- | nvkm/engine/disp/gm107.c | 2 | ||||
-rw-r--r-- | nvkm/engine/disp/hdminve0.c | 86 | ||||
-rw-r--r-- | nvkm/engine/disp/nv50.h | 1 | ||||
-rw-r--r-- | nvkm/engine/disp/nve0.c | 2 | ||||
-rw-r--r-- | nvkm/engine/disp/nvf0.c | 2 |
8 files changed, 93 insertions, 3 deletions
diff --git a/drm/Kbuild b/drm/Kbuild index 949a034e5..12c24c8ab 100644 --- a/drm/Kbuild +++ b/drm/Kbuild @@ -261,6 +261,7 @@ nouveau-y += core/engine/disp/hdanvd0.o nouveau-y += core/engine/disp/hdminv84.o nouveau-y += core/engine/disp/hdminva3.o nouveau-y += core/engine/disp/hdminvd0.o +nouveau-y += core/engine/disp/hdminve0.o nouveau-y += core/engine/disp/piornv50.o nouveau-y += core/engine/disp/sornv50.o nouveau-y += core/engine/disp/sornv94.o diff --git a/drm/core/engine/disp/hdminve0.c b/drm/core/engine/disp/hdminve0.c new file mode 120000 index 000000000..0d7b059c5 --- /dev/null +++ b/drm/core/engine/disp/hdminve0.c @@ -0,0 +1 @@ +../../../../nvkm/engine/disp/hdminve0.c
\ No newline at end of file diff --git a/nvkm/engine/disp/Makefile.am b/nvkm/engine/disp/Makefile.am index bd1b911eb..61e88482e 100644 --- a/nvkm/engine/disp/Makefile.am +++ b/nvkm/engine/disp/Makefile.am @@ -21,6 +21,7 @@ libdisp_la_SOURCES = base.c \ hdminv84.c \ hdminva3.c \ hdminvd0.c \ + hdminve0.c \ piornv50.c \ sornv50.c \ sornv94.c \ diff --git a/nvkm/engine/disp/gm107.c b/nvkm/engine/disp/gm107.c index c8e4e60b8..b3df3fe2d 100644 --- a/nvkm/engine/disp/gm107.c +++ b/nvkm/engine/disp/gm107.c @@ -84,7 +84,7 @@ gm107_disp_ctor(struct nouveau_object *parent, struct nouveau_object *engine, priv->dac.sense = nv50_dac_sense; priv->sor.power = nv50_sor_power; priv->sor.hda_eld = nvd0_hda_eld; - priv->sor.hdmi = nvd0_hdmi_ctrl; + priv->sor.hdmi = nve0_hdmi_ctrl; return 0; } diff --git a/nvkm/engine/disp/hdminve0.c b/nvkm/engine/disp/hdminve0.c new file mode 100644 index 000000000..7bf632a3c --- /dev/null +++ b/nvkm/engine/disp/hdminve0.c @@ -0,0 +1,86 @@ +/* + * Copyright 2014 Red Hat Inc. + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR + * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, + * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + * OTHER DEALINGS IN THE SOFTWARE. + * + * Authors: Ben Skeggs + */ + +#include <core/client.h> +#include <nvif/unpack.h> +#include <nvif/class.h> + +#include "nv50.h" + +int +nve0_hdmi_ctrl(NV50_DISP_MTHD_V1) +{ + const u32 hoff = (head * 0x800); + const u32 hdmi = (head * 0x400); + union { + struct nv50_disp_sor_hdmi_pwr_v0 v0; + } *args = data; + u32 ctrl; + int ret; + + nv_ioctl(object, "disp sor hdmi ctrl size %d\n", size); + if (nvif_unpack(args->v0, 0, 0, false)) { + nv_ioctl(object, "disp sor hdmi ctrl vers %d state %d " + "max_ac_packet %d rekey %d\n", + args->v0.version, args->v0.state, + args->v0.max_ac_packet, args->v0.rekey); + if (args->v0.max_ac_packet > 0x1f || args->v0.rekey > 0x7f) + return -EINVAL; + ctrl = 0x40000000 * !!args->v0.state; + ctrl |= args->v0.max_ac_packet << 16; + ctrl |= args->v0.rekey; + } else + return ret; + + if (!(ctrl & 0x40000000)) { + nv_mask(priv, 0x616798 + hoff, 0x40000000, 0x00000000); + nv_mask(priv, 0x6900c0 + hdmi, 0x00000001, 0x00000000); + nv_mask(priv, 0x690000 + hdmi, 0x00000001, 0x00000000); + return 0; + } + + /* AVI InfoFrame */ + nv_mask(priv, 0x690000 + hdmi, 0x00000001, 0x00000000); + nv_wr32(priv, 0x690008 + hdmi, 0x000d0282); + nv_wr32(priv, 0x69000c + hdmi, 0x0000006f); + nv_wr32(priv, 0x690010 + hdmi, 0x00000000); + nv_wr32(priv, 0x690014 + hdmi, 0x00000000); + nv_wr32(priv, 0x690018 + hdmi, 0x00000000); + nv_mask(priv, 0x690000 + hdmi, 0x00000001, 0x00000001); + + /* ??? InfoFrame? */ + nv_mask(priv, 0x6900c0 + hdmi, 0x00000001, 0x00000000); + nv_wr32(priv, 0x6900cc + hdmi, 0x00000010); + nv_mask(priv, 0x6900c0 + hdmi, 0x00000001, 0x00000001); + + /* ??? */ + nv_wr32(priv, 0x690080 + hdmi, 0x82000000); + + /* HDMI_CTRL */ + nv_mask(priv, 0x616798 + hoff, 0x401f007f, ctrl); + + /* NFI, audio doesn't work without it though.. */ + nv_mask(priv, 0x616548 + hoff, 0x00000070, 0x00000000); + return 0; +} diff --git a/nvkm/engine/disp/nv50.h b/nvkm/engine/disp/nv50.h index a288a7f45..5279feefe 100644 --- a/nvkm/engine/disp/nv50.h +++ b/nvkm/engine/disp/nv50.h @@ -77,6 +77,7 @@ int nvd0_hda_eld(NV50_DISP_MTHD_V1); int nv84_hdmi_ctrl(NV50_DISP_MTHD_V1); int nva3_hdmi_ctrl(NV50_DISP_MTHD_V1); int nvd0_hdmi_ctrl(NV50_DISP_MTHD_V1); +int nve0_hdmi_ctrl(NV50_DISP_MTHD_V1); int nv50_sor_power(NV50_DISP_MTHD_V1); diff --git a/nvkm/engine/disp/nve0.c b/nvkm/engine/disp/nve0.c index 3ed47e187..db144b2cf 100644 --- a/nvkm/engine/disp/nve0.c +++ b/nvkm/engine/disp/nve0.c @@ -249,7 +249,7 @@ nve0_disp_ctor(struct nouveau_object *parent, struct nouveau_object *engine, priv->dac.sense = nv50_dac_sense; priv->sor.power = nv50_sor_power; priv->sor.hda_eld = nvd0_hda_eld; - priv->sor.hdmi = nvd0_hdmi_ctrl; + priv->sor.hdmi = nve0_hdmi_ctrl; return 0; } diff --git a/nvkm/engine/disp/nvf0.c b/nvkm/engine/disp/nvf0.c index 07f337695..402d7d67d 100644 --- a/nvkm/engine/disp/nvf0.c +++ b/nvkm/engine/disp/nvf0.c @@ -84,7 +84,7 @@ nvf0_disp_ctor(struct nouveau_object *parent, struct nouveau_object *engine, priv->dac.sense = nv50_dac_sense; priv->sor.power = nv50_sor_power; priv->sor.hda_eld = nvd0_hda_eld; - priv->sor.hdmi = nvd0_hdmi_ctrl; + priv->sor.hdmi = nve0_hdmi_ctrl; return 0; } |