]>
Commit | Line | Data |
---|---|---|
b97bf3fd | 1 | /* |
5b06c85c | 2 | * net/tipc/subscr.c: TIPC network topology service |
c4307285 | 3 | * |
df79d040 | 4 | * Copyright (c) 2000-2017, Ericsson AB |
13a2e898 | 5 | * Copyright (c) 2005-2007, 2010-2013, Wind River Systems |
b97bf3fd PL |
6 | * All rights reserved. |
7 | * | |
9ea1fd3c | 8 | * Redistribution and use in source and binary forms, with or without |
b97bf3fd PL |
9 | * modification, are permitted provided that the following conditions are met: |
10 | * | |
9ea1fd3c PL |
11 | * 1. Redistributions of source code must retain the above copyright |
12 | * notice, this list of conditions and the following disclaimer. | |
13 | * 2. Redistributions in binary form must reproduce the above copyright | |
14 | * notice, this list of conditions and the following disclaimer in the | |
15 | * documentation and/or other materials provided with the distribution. | |
16 | * 3. Neither the names of the copyright holders nor the names of its | |
17 | * contributors may be used to endorse or promote products derived from | |
18 | * this software without specific prior written permission. | |
b97bf3fd | 19 | * |
9ea1fd3c PL |
20 | * Alternatively, this software may be distributed under the terms of the |
21 | * GNU General Public License ("GPL") version 2 as published by the Free | |
22 | * Software Foundation. | |
23 | * | |
24 | * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | |
25 | * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
26 | * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | |
27 | * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE | |
28 | * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | |
29 | * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | |
30 | * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | |
31 | * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | |
32 | * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | |
33 | * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | |
b97bf3fd PL |
34 | * POSSIBILITY OF SUCH DAMAGE. |
35 | */ | |
36 | ||
37 | #include "core.h" | |
b97bf3fd | 38 | #include "name_table.h" |
5b06c85c | 39 | #include "subscr.h" |
b97bf3fd | 40 | |
da0a75e8 JM |
41 | static void tipc_sub_send_event(struct tipc_subscription *sub, |
42 | u32 found_lower, u32 found_upper, | |
43 | u32 event, u32 port, u32 node) | |
b97bf3fd | 44 | { |
414574a0 | 45 | struct tipc_event *evt = &sub->evt; |
b97bf3fd | 46 | |
df79d040 JM |
47 | if (sub->inactive) |
48 | return; | |
8985ecc7 JM |
49 | tipc_evt_write(evt, event, event); |
50 | tipc_evt_write(evt, found_lower, found_lower); | |
51 | tipc_evt_write(evt, found_upper, found_upper); | |
52 | tipc_evt_write(evt, port.ref, port); | |
53 | tipc_evt_write(evt, port.node, node); | |
026321c6 | 54 | tipc_topsrv_queue_evt(sub->net, sub->conid, event, evt); |
b97bf3fd PL |
55 | } |
56 | ||
57 | /** | |
da0a75e8 | 58 | * tipc_sub_check_overlap - test for subscription overlap with the |
57f1d186 | 59 | * given values |
b97bf3fd PL |
60 | * |
61 | * Returns 1 if there is overlap, otherwise 0. | |
62 | */ | |
da0a75e8 JM |
63 | int tipc_sub_check_overlap(struct tipc_name_seq *seq, u32 found_lower, |
64 | u32 found_upper) | |
b97bf3fd | 65 | { |
a4273c73 PB |
66 | if (found_lower < seq->lower) |
67 | found_lower = seq->lower; | |
68 | if (found_upper > seq->upper) | |
69 | found_upper = seq->upper; | |
b97bf3fd PL |
70 | if (found_lower > found_upper) |
71 | return 0; | |
72 | return 1; | |
73 | } | |
74 | ||
da0a75e8 JM |
75 | void tipc_sub_report_overlap(struct tipc_subscription *sub, |
76 | u32 found_lower, u32 found_upper, | |
77 | u32 event, u32 port, u32 node, | |
78 | u32 scope, int must) | |
a4273c73 | 79 | { |
8985ecc7 JM |
80 | struct tipc_subscr *s = &sub->evt.s; |
81 | u32 filter = tipc_sub_read(s, filter); | |
da0a75e8 | 82 | struct tipc_name_seq seq; |
8985ecc7 JM |
83 | |
84 | seq.type = tipc_sub_read(s, seq.type); | |
85 | seq.lower = tipc_sub_read(s, seq.lower); | |
86 | seq.upper = tipc_sub_read(s, seq.upper); | |
a4273c73 | 87 | |
da0a75e8 | 88 | if (!tipc_sub_check_overlap(&seq, found_lower, found_upper)) |
b97bf3fd | 89 | return; |
8985ecc7 | 90 | |
232d07b7 JM |
91 | if (!must && !(filter & TIPC_SUB_PORTS)) |
92 | return; | |
93 | if (filter & TIPC_SUB_CLUSTER_SCOPE && scope == TIPC_NODE_SCOPE) | |
94 | return; | |
95 | if (filter & TIPC_SUB_NODE_SCOPE && scope != TIPC_NODE_SCOPE) | |
b97bf3fd | 96 | return; |
df79d040 | 97 | spin_lock(&sub->lock); |
da0a75e8 JM |
98 | tipc_sub_send_event(sub, found_lower, found_upper, |
99 | event, port, node); | |
df79d040 | 100 | spin_unlock(&sub->lock); |
b97bf3fd PL |
101 | } |
102 | ||
da0a75e8 | 103 | static void tipc_sub_timeout(struct timer_list *t) |
b97bf3fd | 104 | { |
31b102bb | 105 | struct tipc_subscription *sub = from_timer(sub, t, timer); |
df79d040 | 106 | struct tipc_subscr *s = &sub->evt.s; |
28353e7f | 107 | |
df79d040 | 108 | spin_lock(&sub->lock); |
da0a75e8 JM |
109 | tipc_sub_send_event(sub, s->seq.lower, s->seq.upper, |
110 | TIPC_SUBSCR_TIMEOUT, 0, 0); | |
df79d040 JM |
111 | sub->inactive = true; |
112 | spin_unlock(&sub->lock); | |
eb409460 LC |
113 | } |
114 | ||
da0a75e8 | 115 | static void tipc_sub_kref_release(struct kref *kref) |
d094c4d5 | 116 | { |
5c45ab24 | 117 | kfree(container_of(kref, struct tipc_subscription, kref)); |
d094c4d5 PB |
118 | } |
119 | ||
da0a75e8 | 120 | void tipc_sub_put(struct tipc_subscription *subscription) |
d094c4d5 | 121 | { |
da0a75e8 | 122 | kref_put(&subscription->kref, tipc_sub_kref_release); |
d094c4d5 PB |
123 | } |
124 | ||
da0a75e8 | 125 | void tipc_sub_get(struct tipc_subscription *subscription) |
d094c4d5 PB |
126 | { |
127 | kref_get(&subscription->kref); | |
128 | } | |
129 | ||
5c45ab24 | 130 | struct tipc_subscription *tipc_sub_subscribe(struct net *net, |
242e82cc JM |
131 | struct tipc_subscr *s, |
132 | int conid) | |
a62fbcce | 133 | { |
8985ecc7 | 134 | u32 filter = tipc_sub_read(s, filter); |
da0a75e8 | 135 | struct tipc_subscription *sub; |
242e82cc | 136 | u32 timeout; |
eb409460 | 137 | |
242e82cc JM |
138 | if ((filter & TIPC_SUB_PORTS && filter & TIPC_SUB_SERVICE) || |
139 | (tipc_sub_read(s, seq.lower) > tipc_sub_read(s, seq.upper))) { | |
140 | pr_warn("Subscription rejected, illegal request\n"); | |
7c13c622 | 141 | return NULL; |
b97bf3fd | 142 | } |
28353e7f | 143 | sub = kmalloc(sizeof(*sub), GFP_ATOMIC); |
a10bd924 | 144 | if (!sub) { |
2cf8aa19 | 145 | pr_warn("Subscription rejected, no memory\n"); |
7c13c622 | 146 | return NULL; |
b97bf3fd | 147 | } |
b714295a JM |
148 | INIT_LIST_HEAD(&sub->service_list); |
149 | INIT_LIST_HEAD(&sub->sub_list); | |
5c45ab24 | 150 | sub->net = net; |
df79d040 JM |
151 | sub->conid = conid; |
152 | sub->inactive = false; | |
57f1d186 | 153 | memcpy(&sub->evt.s, s, sizeof(*s)); |
df79d040 | 154 | spin_lock_init(&sub->lock); |
d094c4d5 | 155 | kref_init(&sub->kref); |
8985ecc7 | 156 | tipc_nametbl_subscribe(sub); |
da0a75e8 | 157 | timer_setup(&sub->timer, tipc_sub_timeout, 0); |
8985ecc7 | 158 | timeout = tipc_sub_read(&sub->evt.s, timeout); |
d094c4d5 PB |
159 | if (timeout != TIPC_WAIT_FOREVER) |
160 | mod_timer(&sub->timer, jiffies + msecs_to_jiffies(timeout)); | |
df79d040 | 161 | return sub; |
b97bf3fd PL |
162 | } |
163 | ||
242e82cc | 164 | void tipc_sub_unsubscribe(struct tipc_subscription *sub) |
b97bf3fd | 165 | { |
df79d040 JM |
166 | tipc_nametbl_unsubscribe(sub); |
167 | if (sub->evt.s.timeout != TIPC_WAIT_FOREVER) | |
168 | del_timer_sync(&sub->timer); | |
da0a75e8 JM |
169 | list_del(&sub->sub_list); |
170 | tipc_sub_put(sub); | |
b97bf3fd | 171 | } |