-
Notifications
You must be signed in to change notification settings - Fork 41
/
Copy pathengine.go
264 lines (221 loc) · 8.15 KB
/
engine.go
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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
/*****************************************************************************
* QuantCup 1: Price-Time Matching Engine
*
* Submitted by: voyager (Go port by [email protected])
*
* Design Overview:
* In this implementation, the limit order book is represented using
* a flat linear array (pricePoints), indexed by the numeric price value.
* Each entry in this array corresponds to a specific price point and holds
* an instance of struct pricePoint. This data structure maintains a list
* of outstanding buy/sell orders at the respective price. Each outstanding
* limit order is represented by an instance of struct orderBookEntry.
*
* askMin and bidMax are global variables that maintain starting points,
* at which the matching algorithm initiates its search.
* askMin holds the lowest price that contains at least one outstanding
* sell order. Analogously, bidMax represents the maximum price point that
* contains at least one outstanding buy order.
*
* When a Buy order arrives, we search the book for outstanding Sell orders
* that cross with the incoming order. We start the search at askMin and
* proceed upwards, incrementing askMin until:
* a) The incoming Buy order is filled.
* b) We reach a price point that no longer crosses with the incoming
* limit price (askMin > BuyOrder.price)
* In case b), we create a new orderBookEntry to record the
* remainder of the incoming Buy order and add it to the global order
* book by appending it to the list at pricePoints[BuyOrder.price].
*
* Incoming Sell orders are handled analogously, except that we start at
* bidMax and proceed downwards.
*
* Although this matching algorithm runs in linear time and may, in
* degenerate cases, require scanning a large number of array slots,
* it appears to work reasonably well in practice, at least on the
* simulated data feed (score_feed.h). The vast majority of incoming
* limit orders can be handled by examining no more than two distinct
* price points and no order requires examining more than five price points.
*
* To avoid incurring the costs of dynamic heap-based memory allocation,
* this implementation maintains the full set of orderBookEntry instances
* in a statically-allocated contiguous memory arena (arenaBookEntries).
* Allocating a new entry is simply a matter of bumping up the orderID
* counter (curOrderID) and returning a pointer to arenaBookEntries[curOrderID].
*
* To cancel an order, we simply set its size to zero. Notably, we avoid
* unhooking its orderBookEntry from the list of active orders in order to
* avoid incurring the costs of pointer manipulation and conditional branches.
* This allows us to handle order cancellation requests very efficiently; the
* current implementation requires only one memory store instruction on
* x86_64. During order matching, when we walk the list of outstanding orders,
* we simply skip these zero-sized entries.
*
* The current implementation uses a custom version of strcpy() to copy the string
* fields ("symbol" and "trader") between data structures. This custom version
* has been optimized for the case STRINGLEN=5 and implements loop unrolling
* to eliminate the use of induction variables and conditional branching.
*
* The memory layout of struct orderBookEntry has been optimized for
* efficient cache access.
*****************************************************************************/
package main
type Engine struct {
// Optional callback function that is called when a trade is executed.
Execute func(Execution)
// An array of pricePoint structures representing the entire limit order
// book.
pricePoints [uint(maxPrice) + 1]pricePoint
curOrderID OrderID // Monotonically-increasing orderID.
askMin uint // Minimum Ask price.
bidMax uint // Maximum Bid price.
// Statically-allocated memory arena for order book entries. This data
// structure allows us to avoid the overhead of heap-based memory
// allocation.
bookEntries [maxNumOrders]orderBookEntry
}
// struct orderBookEntry: Describes a single outstanding limit order (Buy or
// Sell).
type orderBookEntry struct {
size Size
next *orderBookEntry
trader string
}
// struct pricePoint: Describes a single price point in the limit order book.
type pricePoint struct {
listHead *orderBookEntry
listTail *orderBookEntry
}
const maxNumOrders uint = 1010000
func (e *Engine) Reset() {
for _, pricePoint := range e.pricePoints {
pricePoint.listHead = nil
pricePoint.listTail = nil
}
for _, bookEntry := range e.bookEntries {
bookEntry.size = 0
bookEntry.next = nil
bookEntry.trader = ""
}
e.curOrderID = 0
e.askMin = uint(maxPrice) + 1
e.bidMax = uint(minPrice) - 1
}
// Process an incoming limit order.
func (e *Engine) Limit(order Order) OrderID {
var price Price = order.price
var orderSize Size = order.size
if order.side == Bid { // Buy order.
// Look for outstanding sell orders that cross with the incoming order.
if uint(price) >= e.askMin {
ppEntry := &e.pricePoints[e.askMin]
for {
bookEntry := ppEntry.listHead
for bookEntry != nil {
if bookEntry.size < orderSize {
execute(e.Execute, order.symbol, order.trader, bookEntry.trader, price, bookEntry.size)
orderSize -= bookEntry.size
bookEntry = bookEntry.next
} else {
execute(e.Execute, order.symbol, order.trader, bookEntry.trader, price, orderSize)
if bookEntry.size > orderSize {
bookEntry.size -= orderSize
} else {
bookEntry = bookEntry.next
}
ppEntry.listHead = bookEntry
e.curOrderID++
return e.curOrderID
}
}
// We have exhausted all orders at the askMin price point. Move
// on to the next price level.
ppEntry.listHead = nil
e.askMin++
ppEntry = &e.pricePoints[e.askMin]
if uint(price) < e.askMin {
break
}
}
}
e.curOrderID++
entry := &e.bookEntries[e.curOrderID]
entry.size = orderSize
entry.trader = order.trader
ppInsertOrder(&e.pricePoints[price], entry)
if e.bidMax < uint(price) {
e.bidMax = uint(price)
}
return e.curOrderID
} else { // Sell order.
// Look for outstanding Buy orders that cross with the incoming order.
if uint(price) <= e.bidMax {
ppEntry := &e.pricePoints[e.bidMax]
for {
bookEntry := ppEntry.listHead
for bookEntry != nil {
if bookEntry.size < orderSize {
execute(e.Execute, order.symbol, bookEntry.trader, order.trader, price, bookEntry.size)
orderSize -= bookEntry.size
bookEntry = bookEntry.next
} else {
execute(e.Execute, order.symbol, bookEntry.trader, order.trader, price, orderSize)
if bookEntry.size > orderSize {
bookEntry.size -= orderSize
} else {
bookEntry = bookEntry.next
}
ppEntry.listHead = bookEntry
e.curOrderID++
return e.curOrderID
}
}
// We have exhausted all orders at the bidMax price point. Move
// on to the next price level.
ppEntry.listHead = nil
e.bidMax--
ppEntry = &e.pricePoints[e.bidMax]
if uint(price) > e.bidMax {
break
}
}
}
e.curOrderID++
entry := &e.bookEntries[e.curOrderID]
entry.size = orderSize
entry.trader = order.trader
ppInsertOrder(&e.pricePoints[price], entry)
if e.askMin > uint(price) {
e.askMin = uint(price)
}
return e.curOrderID
}
}
func (e *Engine) Cancel(orderID OrderID) {
e.bookEntries[orderID].size = 0
}
// Report trade execution.
func execute(hook func(Execution), symbol, buyTrader, sellTrader string, price Price, size Size) {
if hook == nil {
return // No callback defined.
}
if size == 0 {
return // Skip orders that have been cancelled.
}
var exec Execution = Execution{symbol: symbol, price: price, size: size}
exec.side = Bid
exec.trader = buyTrader
hook(exec) // Report the buy-side trade.
exec.side = Ask
exec.trader = sellTrader
hook(exec) // Report the sell-side trade.
}
// Insert a new order book entry at the tail of the price point list.
func ppInsertOrder(ppEntry *pricePoint, entry *orderBookEntry) {
if ppEntry.listHead != nil {
ppEntry.listTail.next = entry
} else {
ppEntry.listHead = entry
}
ppEntry.listTail = entry
}