1 /*******************************************************************************
2 * Copyright (c) 2012 Ericsson
4 * All rights reserved. This program and the accompanying materials are made
5 * available under the terms of the Eclipse Public License v1.0 which
6 * accompanies this distribution, and is available at
7 * http://www.eclipse.org/legal/epl-v10.html
9 * Contributors: Matthew Khouzam - Initial API and implementation
10 *******************************************************************************/
11 package org
.eclipse
.linuxtools
.tmf
.core
.ctfadaptor
;
13 import java
.util
.ArrayList
;
14 import java
.util
.HashMap
;
15 import java
.util
.Random
;
18 * Ctf Iterator Manager, allows mapping of iterators (a limited resource) to
19 * contexts (many many resources).
21 * @author Matthew Khouzam
25 public abstract class CtfIteratorManager
{
27 * A side note synchronized works on the whole object, Therefore add and
28 * remove will be thread safe.
32 * The map of traces to trace managers.
34 private static HashMap
<CtfTmfTrace
, CtfTraceManager
> map
= new HashMap
<CtfTmfTrace
, CtfTraceManager
>();
37 * Registers a trace to the iterator manager, the trace can now get
41 * the trace to register.
43 public static synchronized void addTrace(final CtfTmfTrace trace
) {
44 map
.put(trace
, new CtfTraceManager(trace
));
48 * Removes a trace to the iterator manager.
51 * the trace to register.
53 public static synchronized void removeTrace(final CtfTmfTrace trace
) {
54 CtfTraceManager mgr
= map
.remove(trace
);
61 * Get an iterator for a given trace and context.
67 * @return the iterator
69 public static synchronized CtfIterator
getIterator(final CtfTmfTrace trace
,
70 final CtfTmfLightweightContext ctx
) {
71 return map
.get(trace
).getIterator(ctx
);
78 * @author Matthew Khouzam
80 class CtfTraceManager
{
82 * Cache size. Under 1023 on linux32 systems. Number of file handles
85 private final static int MAX_SIZE
= 100;
87 * The map of the cache.
89 private final HashMap
<CtfTmfLightweightContext
, CtfIterator
> fMap
;
91 * An array pointing to the same cache. this allows fast "random" accesses.
93 private final ArrayList
<CtfTmfLightweightContext
> fRandomAccess
;
97 private final CtfTmfTrace fTrace
;
99 * Random number generator
101 private final Random fRnd
;
103 public CtfTraceManager(CtfTmfTrace trace
) {
104 fMap
= new HashMap
<CtfTmfLightweightContext
, CtfIterator
>();
105 fRandomAccess
= new ArrayList
<CtfTmfLightweightContext
>();
106 fRnd
= new Random(System
.nanoTime());
111 * This needs explaining: the iterator table is effectively a cache.
112 * Originally the contexts had a 1 to 1 structure with the file handles of a
113 * trace. This failed since there is a limit to how many file handles we can
114 * have opened simultaneously. Then a round-robin scheme was implemented,
115 * this lead up to a two competing contexts syncing up and using the same
116 * file handler, causing horrible slowdowns. Now a random replacement
117 * algorithm is selected. This is the same as used by arm processors, and it
118 * works quite well when many cores so this looks promising for very
119 * multi-threaded systems.
122 * the context to look up
123 * @return the iterator refering to the context
125 public CtfIterator
getIterator(final CtfTmfLightweightContext context
) {
127 * if the element is in the map, we don't need to do anything else.
129 CtfIterator retVal
= fMap
.get(context
);
130 if (retVal
== null) {
132 * Assign an iterator to a context, this means we will need to seek
135 if (fRandomAccess
.size() < MAX_SIZE
) {
137 * if we're not full yet, just add an element.
139 retVal
= fTrace
.createIterator();
140 addElement(context
, retVal
);
144 * if we're full, randomly replace an element
146 retVal
= replaceRandomElement(context
);
148 if (context
.getLocation() != null) {
149 final CtfLocationData location
= (CtfLocationData
) context
.getLocation().getLocationInfo();
150 retVal
.seek(location
);
157 * Add a pair of context and element to the hashmap and the arraylist.
164 private void addElement(final CtfTmfLightweightContext context
,
165 final CtfIterator elem
) {
166 fMap
.put(context
, elem
);
167 fRandomAccess
.add(context
);
171 * Replace a random element
174 * the context to swap in
175 * @return the iterator of the removed elements.
177 private CtfIterator
replaceRandomElement(
178 final CtfTmfLightweightContext context
) {
180 * This needs some explanation too: We need to select a random victim
181 * and remove it. The order of the elements is not important, so instead
182 * of just calling arraylist.remove(element) which has an O(n)
183 * complexity, we pick an random number. The element is swapped out of
184 * the array and removed and replaced in the hashmap.
186 final int size
= fRandomAccess
.size();
187 final int pos
= fRnd
.nextInt(size
);
188 final CtfTmfLightweightContext victim
= fRandomAccess
.get(pos
);
189 fRandomAccess
.set(pos
, context
);
190 final CtfIterator elem
= fMap
.remove(victim
);
191 fMap
.put(context
, elem
);
196 for (CtfIterator iterator
: fMap
.values()) {
200 fRandomAccess
.clear();