View Javadoc
1   /*
2    * Copyright (C) 2008, Shawn O. Pearce <spearce@spearce.org>,
3    * Copyright (C) 2013, Gustaf Lundh <gustaf.lundh@sonymobile.com> and others
4    *
5    * This program and the accompanying materials are made available under the
6    * terms of the Eclipse Distribution License v. 1.0 which is available at
7    * https://www.eclipse.org/org/documents/edl-v10.php.
8    *
9    * SPDX-License-Identifier: BSD-3-Clause
10   */
11  
12  package org.eclipse.jgit.revwalk;
13  
14  import java.io.IOException;
15  
16  import org.eclipse.jgit.errors.IncorrectObjectTypeException;
17  import org.eclipse.jgit.errors.MissingObjectException;
18  
19  /**
20   * A queue of commits sorted by commit time order.
21   */
22  public class DateRevQueue extends AbstractRevQueue {
23  	private static final int REBUILD_INDEX_COUNT = 1000;
24  
25  	private Entry head;
26  
27  	private Entry free;
28  
29  	private int inQueue;
30  
31  	private int sinceLastIndex;
32  
33  	private Entry[] index;
34  
35  	private int first;
36  
37  	private int last = -1;
38  
39  	/** Create an empty date queue. */
40  	public DateRevQueue() {
41  		super(false);
42  	}
43  
44  	DateRevQueue(boolean firstParent) {
45  		super(firstParent);
46  	}
47  
48  	DateRevQueue(Generator s) throws MissingObjectException,
49  			IncorrectObjectTypeException, IOException {
50  		super(s.firstParent);
51  		for (;;) {
52  			final RevCommit c = s.next();
53  			if (c == null)
54  				break;
55  			add(c);
56  		}
57  	}
58  
59  	/** {@inheritDoc} */
60  	@Override
61  	public void add(RevCommit c) {
62  		sinceLastIndex++;
63  		if (++inQueue > REBUILD_INDEX_COUNT
64  				&& sinceLastIndex > REBUILD_INDEX_COUNT)
65  			buildIndex();
66  
67  		Entry q = head;
68  		final long when = c.commitTime;
69  
70  		if (first <= last && index[first].commit.commitTime > when) {
71  			int low = first, high = last;
72  			while (low <= high) {
73  				int mid = (low + high) >>> 1;
74  				int t = index[mid].commit.commitTime;
75  				if (t < when)
76  					high = mid - 1;
77  				else if (t > when)
78  					low = mid + 1;
79  				else {
80  					low = mid - 1;
81  					break;
82  				}
83  			}
84  			low = Math.min(low, high);
85  			while (low > first && when == index[low].commit.commitTime)
86  				--low;
87  			q = index[low];
88  		}
89  
90  		final Entry n = newEntry(c);
91  		if (q == null || (q == head && when > q.commit.commitTime)) {
92  			n.next = q;
93  			head = n;
94  		} else {
95  			Entry p = q.next;
96  			while (p != null && p.commit.commitTime >= when) {
97  				q = p;
98  				p = q.next;
99  			}
100 			n.next = q.next;
101 			q.next = n;
102 		}
103 	}
104 
105 	/** {@inheritDoc} */
106 	@Override
107 	public RevCommit next() {
108 		final Entry q = head;
109 		if (q == null)
110 			return null;
111 
112 		if (index != null && q == index[first])
113 			index[first++] = null;
114 		inQueue--;
115 
116 		head = q.next;
117 		freeEntry(q);
118 		return q.commit;
119 	}
120 
121 	private void buildIndex() {
122 		sinceLastIndex = 0;
123 		first = 0;
124 		index = new Entry[inQueue / 100 + 1];
125 		int qi = 0, ii = 0;
126 		for (Entry q = head; q != null; q = q.next) {
127 			if (++qi % 100 == 0)
128 				index[ii++] = q;
129 		}
130 		last = ii - 1;
131 	}
132 
133 	/**
134 	 * Peek at the next commit, without removing it.
135 	 *
136 	 * @return the next available commit; null if there are no commits left.
137 	 */
138 	public RevCommit peek() {
139 		return head != null ? head.commit : null;
140 	}
141 
142 	/** {@inheritDoc} */
143 	@Override
144 	public void clear() {
145 		head = null;
146 		free = null;
147 		index = null;
148 		inQueue = 0;
149 		sinceLastIndex = 0;
150 		last = -1;
151 	}
152 
153 	@Override
154 	boolean everbodyHasFlag(int f) {
155 		for (Entry q = head; q != null; q = q.next) {
156 			if ((q.commit.flags & f) == 0)
157 				return false;
158 		}
159 		return true;
160 	}
161 
162 	@Override
163 	boolean anybodyHasFlag(int f) {
164 		for (Entry q = head; q != null; q = q.next) {
165 			if ((q.commit.flags & f) != 0)
166 				return true;
167 		}
168 		return false;
169 	}
170 
171 	@Override
172 	int outputType() {
173 		return outputType | SORT_COMMIT_TIME_DESC;
174 	}
175 
176 	/** {@inheritDoc} */
177 	@Override
178 	public String toString() {
179 		final StringBuilder s = new StringBuilder();
180 		for (Entry q = head; q != null; q = q.next)
181 			describe(s, q.commit);
182 		return s.toString();
183 	}
184 
185 	private Entry newEntry(RevCommit c) {
186 		Entry r = free;
187 		if (r == null)
188 			r = new Entry();
189 		else
190 			free = r.next;
191 		r.commit = c;
192 		return r;
193 	}
194 
195 	private void freeEntry(Entry e) {
196 		e.next = free;
197 		free = e;
198 	}
199 
200 	static class Entry {
201 		Entry next;
202 
203 		RevCommit commit;
204 	}
205 }