blob: 3bc13df4b120fd12c9d888a40de7bef5f7ed3372 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * linux/drivers/video/nvidia/nv_of.c
3 *
4 * Copyright 2004 Antonino A. Daplas <adaplas @pol.net>
5 *
6 * Based on rivafb-i2c.c
7 *
8 * This file is subject to the terms and conditions of the GNU General Public
9 * License. See the file COPYING in the main directory of this archive
10 * for more details.
11 */
12
Linus Torvalds1da177e2005-04-16 15:20:36 -070013#include <linux/module.h>
14#include <linux/kernel.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070015#include <linux/delay.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090016#include <linux/gfp.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070017#include <linux/pci.h>
18#include <linux/fb.h>
19
20#include <asm/io.h>
21
22#include <asm/prom.h>
23#include <asm/pci-bridge.h>
24
25#include "nv_type.h"
26#include "nv_local.h"
27#include "nv_proto.h"
28
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080029#include "../edid.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070030
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080031int nvidia_probe_of_connector(struct fb_info *info, int conn, u8 **out_edid)
Linus Torvalds1da177e2005-04-16 15:20:36 -070032{
Alexey Dobriyan2d212472005-09-11 04:01:09 +040033 struct nvidia_par *par = info->par;
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080034 struct device_node *parent, *dp;
Jeremy Kerrb04e3dd2006-07-12 15:40:40 +100035 const unsigned char *pedid = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -070036 static char *propnames[] = {
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080037 "DFP,EDID", "LCD,EDID", "EDID", "EDID1",
38 "EDID,B", "EDID,A", NULL };
Linus Torvalds1da177e2005-04-16 15:20:36 -070039 int i;
40
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080041 parent = pci_device_to_OF_node(par->pci_dev);
42 if (parent == NULL)
43 return -1;
44 if (par->twoHeads) {
Jeremy Kerrb04e3dd2006-07-12 15:40:40 +100045 const char *pname;
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080046 int len;
47
48 for (dp = NULL;
49 (dp = of_get_next_child(parent, dp)) != NULL;) {
Stephen Rothwell40cd3a42007-05-01 13:54:02 +100050 pname = of_get_property(dp, "name", NULL);
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080051 if (!pname)
52 continue;
53 len = strlen(pname);
54 if ((pname[len-1] == 'A' && conn == 1) ||
55 (pname[len-1] == 'B' && conn == 2)) {
56 for (i = 0; propnames[i] != NULL; ++i) {
Stephen Rothwell40cd3a42007-05-01 13:54:02 +100057 pedid = of_get_property(dp,
58 propnames[i], NULL);
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080059 if (pedid != NULL)
60 break;
61 }
62 of_node_put(dp);
63 break;
Linus Torvalds1da177e2005-04-16 15:20:36 -070064 }
65 }
66 }
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080067 if (pedid == NULL) {
68 for (i = 0; propnames[i] != NULL; ++i) {
Stephen Rothwell40cd3a42007-05-01 13:54:02 +100069 pedid = of_get_property(parent, propnames[i], NULL);
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080070 if (pedid != NULL)
71 break;
72 }
73 }
74 if (pedid) {
Alexey Dobriyanbfba7b32006-12-08 02:40:46 -080075 *out_edid = kmemdup(pedid, EDID_LENGTH, GFP_KERNEL);
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080076 if (*out_edid == NULL)
77 return -1;
Benjamin Herrenschmidt85f15032005-11-07 01:00:30 -080078 printk(KERN_DEBUG "nvidiafb: Found OF EDID for head %d\n", conn);
79 return 0;
80 }
81 return -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -070082}