-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathresource.c
114 lines (89 loc) · 2.42 KB
/
resource.c
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
104
105
106
107
108
109
110
111
112
113
114
/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
* Copyright(c) 2021 John Sanpe <[email protected]>
*/
#define MODULE_NAME "resource"
#define pr_fmt(fmt) MODULE_NAME ": " fmt
#include <resource.h>
#include <spinlock.h>
#include <ioops.h>
#include <export.h>
struct resource root_mmio = {
.name = "root mmio",
.start = 0,
.size = -1,
.type = RESOURCE_MMIO,
};
EXPORT_SYMBOL(root_mmio);
struct resource root_pmio = {
.name = "root pmio",
.start = 0,
.size = IO_SPACE_LIMIT,
.type = RESOURCE_PMIO,
};
EXPORT_SYMBOL(root_pmio);
static SPIN_LOCK(resource_lock);
static struct resource *request_resource(struct resource *parent, struct resource *res)
{
struct list_head *head;
struct resource *walk;
if (!res->size)
return ERR_PTR(-EINVAL);
if (res->start < parent->start ||
resource_end(res) > resource_end(parent))
return ERR_PTR(-EADDRNOTAVAIL);
head = &parent->child;
list_for_each_entry(walk, &parent->child, siblings) {
if (walk->start > res->start)
break;
if (resource_end(walk) > resource_end(res))
return walk;
head = &walk->siblings;
}
res->parent = parent;
list_add(head, &res->siblings);
return NULL;
}
static state release_resource(struct resource *res)
{
struct resource *parent = res->parent;
struct resource *walk, *find;
list_for_each_entry(walk, &parent->child, siblings) {
if (walk == res) {
find = walk;
break;
}
}
if (unlikely(!find))
return -ENOENT;
res->parent = NULL;
list_del(&res->siblings);
return -ENOERR;
}
struct resource *resource_request_conflict(struct resource *parent, struct resource *res)
{
struct resource *find;
spin_lock(&resource_lock);
find = request_resource(parent, res);
spin_unlock(&resource_lock);
return find;
}
EXPORT_SYMBOL(resource_request_conflict);
state resource_request(struct resource *parent, struct resource *res)
{
struct resource *conflict;
conflict = resource_request_conflict(parent, res);
if (IS_ERR(conflict))
return PTR_ERR(conflict);
return conflict ? -EBUSY : -ENOERR;
}
EXPORT_SYMBOL(resource_request);
state resource_release(struct resource *res)
{
state retval;
spin_lock(&resource_lock);
retval = release_resource(res);
spin_unlock(&resource_lock);
return retval;
}
EXPORT_SYMBOL(resource_release);