Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* |
| 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 Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 13 | #include <linux/module.h> |
| 14 | #include <linux/kernel.h> |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 15 | #include <linux/delay.h> |
Tejun Heo | 5a0e3ad | 2010-03-24 17:04:11 +0900 | [diff] [blame] | 16 | #include <linux/gfp.h> |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 17 | #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 Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 29 | #include "../edid.h" |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 30 | |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 31 | int nvidia_probe_of_connector(struct fb_info *info, int conn, u8 **out_edid) |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 32 | { |
Alexey Dobriyan | 2d21247 | 2005-09-11 04:01:09 +0400 | [diff] [blame] | 33 | struct nvidia_par *par = info->par; |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 34 | struct device_node *parent, *dp; |
Jeremy Kerr | b04e3dd | 2006-07-12 15:40:40 +1000 | [diff] [blame] | 35 | const unsigned char *pedid = NULL; |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 36 | static char *propnames[] = { |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 37 | "DFP,EDID", "LCD,EDID", "EDID", "EDID1", |
| 38 | "EDID,B", "EDID,A", NULL }; |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 39 | int i; |
| 40 | |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 41 | parent = pci_device_to_OF_node(par->pci_dev); |
| 42 | if (parent == NULL) |
| 43 | return -1; |
| 44 | if (par->twoHeads) { |
Jeremy Kerr | b04e3dd | 2006-07-12 15:40:40 +1000 | [diff] [blame] | 45 | const char *pname; |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 46 | int len; |
| 47 | |
| 48 | for (dp = NULL; |
| 49 | (dp = of_get_next_child(parent, dp)) != NULL;) { |
Stephen Rothwell | 40cd3a4 | 2007-05-01 13:54:02 +1000 | [diff] [blame] | 50 | pname = of_get_property(dp, "name", NULL); |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 51 | 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 Rothwell | 40cd3a4 | 2007-05-01 13:54:02 +1000 | [diff] [blame] | 57 | pedid = of_get_property(dp, |
| 58 | propnames[i], NULL); |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 59 | if (pedid != NULL) |
| 60 | break; |
| 61 | } |
| 62 | of_node_put(dp); |
| 63 | break; |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 64 | } |
| 65 | } |
| 66 | } |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 67 | if (pedid == NULL) { |
| 68 | for (i = 0; propnames[i] != NULL; ++i) { |
Stephen Rothwell | 40cd3a4 | 2007-05-01 13:54:02 +1000 | [diff] [blame] | 69 | pedid = of_get_property(parent, propnames[i], NULL); |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 70 | if (pedid != NULL) |
| 71 | break; |
| 72 | } |
| 73 | } |
| 74 | if (pedid) { |
Alexey Dobriyan | bfba7b3 | 2006-12-08 02:40:46 -0800 | [diff] [blame] | 75 | *out_edid = kmemdup(pedid, EDID_LENGTH, GFP_KERNEL); |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 76 | if (*out_edid == NULL) |
| 77 | return -1; |
Benjamin Herrenschmidt | 85f1503 | 2005-11-07 01:00:30 -0800 | [diff] [blame] | 78 | printk(KERN_DEBUG "nvidiafb: Found OF EDID for head %d\n", conn); |
| 79 | return 0; |
| 80 | } |
| 81 | return -1; |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 82 | } |