summaryrefslogtreecommitdiff
path: root/arch/arm/mach-tegra/kfuse.c
blob: f5de828bb50849fa2a916300463c8cff7d5c5e0a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
/*
 * arch/arm/mach-tegra/kfuse.c
 *
 * Copyright (C) 2010-2011 NVIDIA Corporation.
 *
 * This software is licensed under the terms of the GNU General Public
 * License version 2, as published by the Free Software Foundation, and
 * may be copied, distributed, and modified under those terms.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 */

/* The kfuse block stores downstream and upstream HDCP keys for use by HDMI
 * module.
 */

#include <linux/kernel.h>
#include <linux/io.h>
#include <linux/err.h>
#include <linux/string.h>
#include <linux/delay.h>

#include <mach/iomap.h>
#include <mach/kfuse.h>

#include "apbio.h"

/* register definition */
#define KFUSE_STATE					0x80
#define  KFUSE_STATE_DONE		(1u << 16)
#define  KFUSE_STATE_CRCPASS		(1u << 17)
#define KFUSE_KEYADDR					0x88
#define  KFUSE_KEYADDR_AUTOINC		(1u << 16)
#define KFUSE_KEYS					0x8c

static inline u32 tegra_kfuse_readl(unsigned long offset)
{
	return tegra_apb_readl(TEGRA_KFUSE_BASE + offset);
}

static inline void tegra_kfuse_writel(u32 value, unsigned long offset)
{
	tegra_apb_writel(value, TEGRA_KFUSE_BASE + offset);
}

static int wait_for_done(void)
{
	u32 reg;
	int retries = 50;
	do {
		reg = tegra_kfuse_readl(KFUSE_STATE);
		if (reg & KFUSE_STATE_DONE);
			return 0;
		msleep(10);
	} while(--retries);
	return -ETIMEDOUT;
}

/* read up to KFUSE_DATA_SZ bytes into dest.
 * always starts at the first kfuse.
 */
int tegra_kfuse_read(void *dest, size_t len)
{
	u32 v;
	unsigned cnt;

	if (len > KFUSE_DATA_SZ)
		return -EINVAL;

	tegra_kfuse_writel(KFUSE_KEYADDR_AUTOINC, KFUSE_KEYADDR);
	wait_for_done();

	if ((tegra_kfuse_readl(KFUSE_STATE) & KFUSE_STATE_CRCPASS) == 0) {
		pr_err("kfuse: crc failed\n");
		return -EIO;
	}

	for (cnt = 0; cnt < len; cnt += 4) {
		v = tegra_kfuse_readl(KFUSE_KEYS);
		memcpy(dest + cnt, &v, sizeof v);
	}

	return 0;
}