aboutsummaryrefslogtreecommitdiffstats
path: root/security/smc/rproc_drm.c
blob: b86b0b859354d5d6e38c6b7f0e19959f5f5d3391 (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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/*
 * Copyright (c) 2011 Texas Instruments, Inc.
 * Copyright (c) 2011 Trusted Logic S.A.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * version 2 as published by the Free Software Foundation.
 *
 * 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.
 */

/*
 * This file implements the non-secure rproc and smc interface/integration
 */

#include <linux/types.h>
#include <linux/module.h>

#include "tee_client_api.h"
#include "tf_defs.h"

/* 7B1DD682-1077-4939-9755-B6192C5CC5FD */
#define WVDRM_UUID {0x7B1DD682, 0x1077, 0x4939, \
			{0x97, 0x55, 0xB6, 0x19, 0x2C, 0x5C, 0xC5, 0xFD} }

#define WVDRM_ENTER_SECURE_PLAYBACK	0x00003000

#define WVDRM_EXIT_SECURE_PLAYBACK	0x00003001

enum rproc_drm_s_state {
	RPROC_DRM_SECURE_LEAVE,
	RPROC_DRM_SECURE_ENTER
};

static enum rproc_drm_s_state s_state;

static TEEC_Result rproc_drm_initialize(TEEC_Context *teec_context,
					TEEC_Session *teec_session)
{
	static const TEEC_UUID drm_uuid = WVDRM_UUID;
	static u32 drm_gid = 1019;
	TEEC_Result result;

	result = TEEC_InitializeContext(NULL, teec_context);
	if (result != TEEC_SUCCESS)
		goto exit;

	result = TEEC_OpenSession(teec_context, teec_session, &drm_uuid,
			TEEC_LOGIN_PRIVILEGED, &drm_gid, NULL, NULL);
	if (result != TEEC_SUCCESS)
		TEEC_FinalizeContext(teec_context);

exit:
	return result;
}

static TEEC_Result rproc_drm_finalize(TEEC_Context *teec_context,
					TEEC_Session *teec_session)
{
	TEEC_CloseSession(teec_session);
	TEEC_FinalizeContext(teec_context);
	return TEEC_SUCCESS;
}

static TEEC_Result _rproc_drm_invoke_secure_service(bool enable)
{
	TEEC_Result result;
	TEEC_Operation operation;
	TEEC_Context teec_context;
	TEEC_Session teec_session;
	u32 command;

	result = rproc_drm_initialize(&teec_context, &teec_session);
	if (result != TEEC_SUCCESS)
		goto out;

	operation.paramTypes = TEEC_PARAM_TYPES(TEEC_NONE, TEEC_NONE,
						TEEC_NONE, TEEC_NONE);
	command = (enable ? WVDRM_ENTER_SECURE_PLAYBACK :
				WVDRM_EXIT_SECURE_PLAYBACK);
	result = TEEC_InvokeCommand(&teec_session, command, &operation, NULL);
	rproc_drm_finalize(&teec_context, &teec_session);
out:
	return result;
}

int rproc_drm_invoke_service(bool enable)
{
	int ret;

	if ((s_state == RPROC_DRM_SECURE_ENTER && enable) ||
		(s_state == RPROC_DRM_SECURE_LEAVE && !enable))
		return 0;

	ret = _rproc_drm_invoke_secure_service(enable);
	s_state = (enum rproc_drm_s_state) enable;

	return ret == TEEC_SUCCESS ? 0 : -EACCES;
}
EXPORT_SYMBOL(rproc_drm_invoke_service);