blob: 28dff36e96af39c07ad345219e0e75dc0b9d2b21 [file] [log] [blame]
Matt Rosencrantzf5afcaf2014-06-02 11:31:22 -07001package ipc
2
3import (
Matt Rosencrantz137b8d22014-08-18 09:56:15 -07004 "sync"
5 "time"
6
Jiri Simsa519c5072014-09-17 21:37:57 -07007 "veyron.io/veyron/veyron2"
8 "veyron.io/veyron/veyron2/context"
Matt Rosencrantzf5afcaf2014-06-02 11:31:22 -07009)
10
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -080011const nilRuntimeMessage = "attempting to create a context with a nil runtime"
12
Matt Rosencrantz29147f72014-06-06 12:46:01 -070013// InternalNewContext creates a new context.T. This function should only
Matt Rosencrantzf5afcaf2014-06-02 11:31:22 -070014// be called from within the runtime implementation.
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070015func InternalNewContext(runtime veyron2.Runtime) context.T {
16 if runtime == nil {
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -080017 panic(nilRuntimeMessage)
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070018 }
19 return rootContext{runtime}
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070020}
21
22// cancellable is an interface to cancellable contexts.
23type cancellable interface {
24 // cancel cancels the context and records the given error.
25 cancel(err error)
26 addChild(child cancellable)
27 removeChild(parent cancellable)
28}
29
30// child is an interface that allows you to find the nearest cancelContext ancestor.
31type child interface {
32 parent() context.T
33}
34
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070035// rootContext is an empty root context. It has no deadline or values
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070036// and can't be canceled.
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070037type rootContext struct {
38 runtime veyron2.Runtime
39}
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070040
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070041func (r rootContext) parent() context.T { return nil }
42func (r rootContext) Deadline() (deadline time.Time, ok bool) { return }
43func (r rootContext) Done() <-chan struct{} { return nil }
44func (r rootContext) Err() error { return nil }
45func (r rootContext) Value(key interface{}) interface{} { return nil }
46func (r rootContext) Runtime() interface{} { return r.runtime }
47func (r rootContext) WithCancel() (ctx context.T, cancel context.CancelFunc) {
48 return newCancelContext(r)
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070049}
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070050func (r rootContext) WithDeadline(deadline time.Time) (context.T, context.CancelFunc) {
51 return newDeadlineContext(r, deadline)
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070052}
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070053func (r rootContext) WithTimeout(timeout time.Duration) (context.T, context.CancelFunc) {
54 return newDeadlineContext(r, time.Now().Add(timeout))
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070055}
Matt Rosencrantzbf85d542014-08-22 13:31:14 -070056func (r rootContext) WithValue(key interface{}, val interface{}) context.T {
57 return newValueContext(r, key, val)
Matt Rosencrantz137b8d22014-08-18 09:56:15 -070058}
59
60// A valueContext contains a single key/value mapping.
61type valueContext struct {
62 context.T
63 key, value interface{}
64}
65
66func newValueContext(parent context.T, key, val interface{}) *valueContext {
67 return &valueContext{parent, key, val}
68}
69
70func (v *valueContext) parent() context.T {
71 return v.T
72}
73func (v *valueContext) Value(key interface{}) interface{} {
74 if key == v.key {
75 return v.value
76 }
77 return v.T.Value(key)
78}
79func (v *valueContext) WithCancel() (ctx context.T, cancel context.CancelFunc) {
80 return newCancelContext(v)
81}
82func (v *valueContext) WithDeadline(deadline time.Time) (context.T, context.CancelFunc) {
83 return newDeadlineContext(v, deadline)
84}
85func (v *valueContext) WithTimeout(timeout time.Duration) (context.T, context.CancelFunc) {
86 return newDeadlineContext(v, time.Now().Add(timeout))
87}
88func (v *valueContext) WithValue(key interface{}, val interface{}) context.T {
89 return newValueContext(v, key, val)
90}
91
92// A cancelContext provides a mechanism for cancellation and a
93// done channel that allows it's status to be monitored.
94type cancelContext struct {
95 context.T
96 done chan struct{}
97 err error
98
Jiri Simsa2874e0b2014-11-11 18:34:42 -080099 // children is used to keep track of descendant cancellable
100 // contexts. This is an optimization to prevent excessive
101 // goroutines.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700102 children map[cancellable]bool
103
104 mu sync.Mutex
105}
106
107func newCancelContext(parent context.T) (ctx *cancelContext, cancel context.CancelFunc) {
108 ctx = &cancelContext{
109 T: parent,
110 done: make(chan struct{}),
111 }
112
113 cancel = func() { ctx.cancel(context.Canceled) }
114 if parent.Done() == nil {
115 return
116 }
117
118 if ancestor, nonStandardAncestor := ctx.findCancellableAncestor(); !nonStandardAncestor {
119 if ancestor != nil {
120 ancestor.addChild(ctx)
121 }
122 return
123 }
124
125 // If neither the parent nor the child are canceled then both the
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800126 // parent and the child will leak. Note this will only happen for
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700127 // non-standard implementations of the context.T interface.
128 go func() {
129 select {
130 case <-parent.Done():
131 ctx.cancel(parent.Err())
132 case <-ctx.Done():
133 }
134 }()
135
136 return
137}
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700138
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800139// addChild sets child as a descendant cancellable context. This
140// allows us to propagate cancellations through the context tree.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700141func (c *cancelContext) addChild(child cancellable) {
142 c.mu.Lock()
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700143 if c.err != nil {
144 // If the parent is already canceled, just cancel the child.
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800145 c.mu.Unlock()
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700146 child.cancel(c.err)
147 return
148 }
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800149 defer c.mu.Unlock()
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700150 if c.children == nil {
151 c.children = make(map[cancellable]bool)
152 }
153 c.children[child] = true
154}
155
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800156// removeChild is called by descendant contexts when they are
157// canceled. This prevents old contexts which are no longer relevant
158// from consuming resources.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700159func (c *cancelContext) removeChild(child cancellable) {
160 c.mu.Lock()
161 defer c.mu.Unlock()
162 delete(c.children, child)
163}
164
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800165// cancelChildren cancels all descendant cancellable contexts. This
166// is called during cancel but while mu is NOT held. Children may try
167// to make calls to parents, which would result in a deadlock.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700168func cancelChildren(children map[cancellable]bool, err error) {
169 for child, _ := range children {
170 child.cancel(err)
171 }
172}
173
174// cancel cancels the context, propagates that signal to children,
175// and updates parents.
176func (c *cancelContext) cancel(err error) {
177 if err == nil {
178 panic("Context canceled with nil error.")
179 }
180 c.mu.Lock()
181 // cancelChilren should be called after mu is released.
182 defer cancelChildren(c.children, err)
183 defer c.mu.Unlock()
184 if c.err != nil {
185 return
186 }
187 c.err = err
188 c.children = nil
189 if ancestor, nonStandardAncestor := c.findCancellableAncestor(); !nonStandardAncestor {
190 if ancestor != nil {
191 ancestor.removeChild(c)
192 }
193 }
194 close(c.done)
195}
196
197// findCancelAncestor finds the nearest ancestor that supports cancellation.
198// nonStandardAncestor will be true if we cannot determine if there is a cancellable
199// ancestor due to the presence of an unknown context implementation. In this
200// case ancestor will always be nil.
201func (c *cancelContext) findCancellableAncestor() (ancestor cancellable, nonStandardAncestor bool) {
202 parent := c.T
203 for {
204 if c, ok := parent.(cancellable); ok {
205 return c, false
206 }
207 c, ok := parent.(child)
208 if !ok {
209 return nil, true
210 }
211 parent = c.parent()
212 }
Matt Rosencrantz4d7c4be2014-11-12 16:23:33 -0800213 return nil, false // Unreachable.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700214}
215
216func (c *cancelContext) Done() <-chan struct{} { return c.done }
217func (c *cancelContext) Err() error {
218 c.mu.Lock()
219 defer c.mu.Unlock()
220 return c.err
221}
222func (c *cancelContext) WithCancel() (ctx context.T, cancel context.CancelFunc) {
223 return newCancelContext(c)
224}
225func (c *cancelContext) WithDeadline(deadline time.Time) (context.T, context.CancelFunc) {
226 return newDeadlineContext(c, deadline)
227}
228func (c *cancelContext) WithTimeout(timeout time.Duration) (context.T, context.CancelFunc) {
229 return newDeadlineContext(c, time.Now().Add(timeout))
230}
231func (c *cancelContext) WithValue(key interface{}, val interface{}) context.T {
232 return newValueContext(c, key, val)
233}
234
235// A deadlineContext automatically cancels itself when the deadline is reached.
236type deadlineContext struct {
237 *cancelContext
238 deadline time.Time
239 timer *time.Timer
240}
241
242// newDeadlineContext returns a new deadlineContext.
243func newDeadlineContext(parent context.T, deadline time.Time) (*deadlineContext, context.CancelFunc) {
244 cancel, _ := newCancelContext(parent)
245 ctx := &deadlineContext{
246 cancelContext: cancel,
247 deadline: deadline,
248 }
249 delta := deadline.Sub(time.Now())
250 ctx.timer = time.AfterFunc(delta, func() { cancel.cancel(context.DeadlineExceeded) })
251 return ctx, func() { ctx.cancel(context.Canceled) }
252}
253
Jiri Simsa2874e0b2014-11-11 18:34:42 -0800254// cancel cancels the deadlineContext, forwards the signal to
255// descendants, and notifies parents.
Matt Rosencrantz137b8d22014-08-18 09:56:15 -0700256func (d *deadlineContext) cancel(err error) {
257 d.timer.Stop()
258 d.cancelContext.cancel(err)
259}
260func (d *deadlineContext) Deadline() (deadline time.Time, ok bool) { return d.deadline, true }
261func (d *deadlineContext) WithCancel() (ctx context.T, cancel context.CancelFunc) {
262 return newCancelContext(d)
263}
264func (d *deadlineContext) WithDeadline(deadline time.Time) (context.T, context.CancelFunc) {
265 return newDeadlineContext(d, deadline)
266}
267func (d *deadlineContext) WithTimeout(timeout time.Duration) (context.T, context.CancelFunc) {
268 return newDeadlineContext(d, time.Now().Add(timeout))
269}
270func (d *deadlineContext) WithValue(key interface{}, val interface{}) context.T {
271 return newValueContext(d, key, val)
Matt Rosencrantzf5afcaf2014-06-02 11:31:22 -0700272}