1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.os;
18 
19 import android.annotation.UnsupportedAppUsage;
20 
21 /** @hide */
22 public class Broadcaster
23 {
24     @UnsupportedAppUsage
Broadcaster()25     public Broadcaster()
26     {
27     }
28 
29     /**
30      *  Sign up for notifications about something.
31      *
32      *  When this broadcaster pushes a message with senderWhat in the what field,
33      *  target will be sent a copy of that message with targetWhat in the what field.
34      */
35     @UnsupportedAppUsage
request(int senderWhat, Handler target, int targetWhat)36     public void request(int senderWhat, Handler target, int targetWhat)
37     {
38         synchronized (this) {
39             Registration r = null;
40             if (mReg == null) {
41                 r = new Registration();
42                 r.senderWhat = senderWhat;
43                 r.targets = new Handler[1];
44                 r.targetWhats = new int[1];
45                 r.targets[0] = target;
46                 r.targetWhats[0] = targetWhat;
47                 mReg = r;
48                 r.next = r;
49                 r.prev = r;
50             } else {
51                 // find its place in the map
52                 Registration start = mReg;
53                 r = start;
54                 do {
55                     if (r.senderWhat >= senderWhat) {
56                         break;
57                     }
58                     r = r.next;
59                 } while (r != start);
60                 int n;
61                 if (r.senderWhat != senderWhat) {
62                     // we didn't find a senderWhat match, but r is right
63                     // after where it goes
64                     Registration reg = new Registration();
65                     reg.senderWhat = senderWhat;
66                     reg.targets = new Handler[1];
67                     reg.targetWhats = new int[1];
68                     reg.next = r;
69                     reg.prev = r.prev;
70                     r.prev.next = reg;
71                     r.prev = reg;
72 
73                     if (r == mReg && r.senderWhat > reg.senderWhat) {
74                         mReg = reg;
75                     }
76 
77                     r = reg;
78                     n = 0;
79                 } else {
80                     n = r.targets.length;
81                     Handler[] oldTargets = r.targets;
82                     int[] oldWhats = r.targetWhats;
83                     // check for duplicates, and don't do it if we are dup.
84                     for (int i=0; i<n; i++) {
85                         if (oldTargets[i] == target && oldWhats[i] == targetWhat) {
86                             return;
87                         }
88                     }
89                     r.targets = new Handler[n+1];
90                     System.arraycopy(oldTargets, 0, r.targets, 0, n);
91                     r.targetWhats = new int[n+1];
92                     System.arraycopy(oldWhats, 0, r.targetWhats, 0, n);
93                 }
94                 r.targets[n] = target;
95                 r.targetWhats[n] = targetWhat;
96             }
97         }
98     }
99 
100     /**
101      * Unregister for notifications for this senderWhat/target/targetWhat tuple.
102      */
103     @UnsupportedAppUsage
cancelRequest(int senderWhat, Handler target, int targetWhat)104     public void cancelRequest(int senderWhat, Handler target, int targetWhat)
105     {
106         synchronized (this) {
107             Registration start = mReg;
108             Registration r = start;
109 
110             if (r == null) {
111                 return;
112             }
113 
114             do {
115                 if (r.senderWhat >= senderWhat) {
116                     break;
117                 }
118                 r = r.next;
119             } while (r != start);
120 
121             if (r.senderWhat == senderWhat) {
122                 Handler[] targets = r.targets;
123                 int[] whats = r.targetWhats;
124                 int oldLen = targets.length;
125                 for (int i=0; i<oldLen; i++) {
126                     if (targets[i] == target && whats[i] == targetWhat) {
127                         r.targets = new Handler[oldLen-1];
128                         r.targetWhats = new int[oldLen-1];
129                         if (i > 0) {
130                             System.arraycopy(targets, 0, r.targets, 0, i);
131                             System.arraycopy(whats, 0, r.targetWhats, 0, i);
132                         }
133 
134                         int remainingLen = oldLen-i-1;
135                         if (remainingLen != 0) {
136                             System.arraycopy(targets, i+1, r.targets, i,
137                                     remainingLen);
138                             System.arraycopy(whats, i+1, r.targetWhats, i,
139                                     remainingLen);
140                         }
141                         break;
142                     }
143                 }
144             }
145         }
146     }
147 
148     /**
149      * For debugging purposes, print the registrations to System.out
150      */
dumpRegistrations()151     public void dumpRegistrations()
152     {
153         synchronized (this) {
154             Registration start = mReg;
155             System.out.println("Broadcaster " + this + " {");
156             if (start != null) {
157                 Registration r = start;
158                 do {
159                     System.out.println("    senderWhat=" + r.senderWhat);
160                     int n = r.targets.length;
161                     for (int i=0; i<n; i++) {
162                         System.out.println("        [" + r.targetWhats[i]
163                                         + "] " + r.targets[i]);
164                     }
165                     r = r.next;
166                 } while (r != start);
167             }
168             System.out.println("}");
169         }
170     }
171 
172     /**
173      * Send out msg.  Anyone who has registered via the request() method will be
174      * sent the message.
175      */
176     @UnsupportedAppUsage
broadcast(Message msg)177     public void broadcast(Message msg)
178     {
179         synchronized (this) {
180             if (mReg == null) {
181                 return;
182             }
183 
184             int senderWhat = msg.what;
185             Registration start = mReg;
186             Registration r = start;
187             do {
188                 if (r.senderWhat >= senderWhat) {
189                     break;
190                 }
191                 r = r.next;
192             } while (r != start);
193             if (r.senderWhat == senderWhat) {
194                 Handler[] targets = r.targets;
195                 int[] whats = r.targetWhats;
196                 int n = targets.length;
197                 for (int i=0; i<n; i++) {
198                     Handler target = targets[i];
199                     Message m = Message.obtain();
200                     m.copyFrom(msg);
201                     m.what = whats[i];
202                     target.sendMessage(m);
203                 }
204             }
205         }
206     }
207 
208     private class Registration
209     {
210         Registration next;
211         Registration prev;
212 
213         int senderWhat;
214         Handler[] targets;
215         int[] targetWhats;
216     }
217     private Registration mReg;
218 }
219