1 // token.h -- lock tokens for gold   -*- C++ -*-
2 
3 // Copyright (C) 2006-2014 Free Software Foundation, Inc.
4 // Written by Ian Lance Taylor <iant@google.com>.
5 
6 // This file is part of gold.
7 
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 3 of the License, or
11 // (at your option) any later version.
12 
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the Free Software
20 // Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21 // MA 02110-1301, USA.
22 
23 #ifndef GOLD_TOKEN_H
24 #define GOLD_TOKEN_H
25 
26 namespace gold
27 {
28 
29 class Condvar;
30 class Task;
31 
32 // A list of Tasks, managed through the next_locked_ field in the
33 // class Task.  We define this class here because we need it in
34 // Task_token.
35 
36 class Task_list
37 {
38  public:
Task_list()39   Task_list()
40     : head_(NULL), tail_(NULL)
41   { }
42 
~Task_list()43   ~Task_list()
44   { gold_assert(this->head_ == NULL && this->tail_ == NULL); }
45 
46   // Return whether the list is empty.
47   bool
empty()48   empty() const
49   { return this->head_ == NULL; }
50 
51   // Add T to the head of the list.
52   void
53   push_front(Task* t);
54 
55   // Add T to the end of the list.
56   void
57   push_back(Task* t);
58 
59   // Remove the first Task on the list and return it.  Return NULL if
60   // the list is empty.
61   Task*
62   pop_front();
63 
64  private:
65   // The start of the list.  NULL if the list is empty.
66   Task* head_;
67   // The end of the list.  NULL if the list is empty.
68   Task* tail_;
69 };
70 
71 // We support two basic types of locks, which are both implemented
72 // using the single class Task_token.
73 
74 // A write lock may be held by a single Task at a time.  This is used
75 // to control access to a single shared resource such as an Object.
76 
77 // A blocker is used to indicate that a Task A must be run after some
78 // set of Tasks B.  For each of the Tasks B, we increment the blocker
79 // when the Task is created, and decrement it when the Task is
80 // completed.  When the count goes to 0, the task A is ready to run.
81 
82 // There are no shared read locks.  We always read and write objects
83 // in predictable patterns.  The purpose of the locks is to permit
84 // some flexibility for the threading system, for cases where the
85 // execution order does not matter.
86 
87 // These tokens are only manipulated when the workqueue lock is held
88 // or when they are first created.  They do not require any locking
89 // themselves.
90 
91 class Task_token
92 {
93  public:
Task_token(bool is_blocker)94   Task_token(bool is_blocker)
95     : is_blocker_(is_blocker), blockers_(0), writer_(NULL), waiting_()
96   { }
97 
~Task_token()98   ~Task_token()
99   {
100     gold_assert(this->blockers_ == 0);
101     gold_assert(this->writer_ == NULL);
102   }
103 
104   // Return whether this is a blocker.
105   bool
is_blocker()106   is_blocker() const
107   { return this->is_blocker_; }
108 
109   // A write lock token uses these methods.
110 
111   // Is the token writable?
112   bool
is_writable()113   is_writable() const
114   {
115     gold_assert(!this->is_blocker_);
116     return this->writer_ == NULL;
117   }
118 
119   // Add the task as the token's writer (there may only be one
120   // writer).
121   void
add_writer(const Task * t)122   add_writer(const Task* t)
123   {
124     gold_assert(!this->is_blocker_ && this->writer_ == NULL);
125     this->writer_ = t;
126   }
127 
128   // Remove the task as the token's writer.
129   void
remove_writer(const Task * t)130   remove_writer(const Task* t)
131   {
132     gold_assert(!this->is_blocker_ && this->writer_ == t);
133     this->writer_ = NULL;
134   }
135 
136   // A blocker token uses these methods.
137 
138   // Add a blocker to the token.
139   void
add_blocker()140   add_blocker()
141   {
142     gold_assert(this->is_blocker_);
143     ++this->blockers_;
144     this->writer_ = NULL;
145   }
146 
147   // Add some number of blockers to the token.
148   void
add_blockers(int c)149   add_blockers(int c)
150   {
151     gold_assert(this->is_blocker_);
152     this->blockers_ += c;
153     this->writer_ = NULL;
154   }
155 
156   // Remove a blocker from the token.  Returns true if block count
157   // drops to zero.
158   bool
remove_blocker()159   remove_blocker()
160   {
161     gold_assert(this->is_blocker_ && this->blockers_ > 0);
162     --this->blockers_;
163     this->writer_ = NULL;
164     return this->blockers_ == 0;
165   }
166 
167   // Is the token currently blocked?
168   bool
is_blocked()169   is_blocked() const
170   {
171     gold_assert(this->is_blocker_);
172     return this->blockers_ > 0;
173   }
174 
175   // Both blocker and write lock tokens use these methods.
176 
177   // Add T to the list of tasks waiting for this token to be released.
178   void
add_waiting(Task * t)179   add_waiting(Task* t)
180   { this->waiting_.push_back(t); }
181 
182   // Add T to the front of the list of tasks waiting for this token to
183   // be released.
184   void
add_waiting_front(Task * t)185   add_waiting_front(Task* t)
186   { this->waiting_.push_front(t); }
187 
188   // Remove the first Task waiting for this token to be released, and
189   // return it.  Return NULL if no Tasks are waiting.
190   Task*
remove_first_waiting()191   remove_first_waiting()
192   { return this->waiting_.pop_front(); }
193 
194  private:
195   // It makes no sense to copy these.
196   Task_token(const Task_token&);
197   Task_token& operator=(const Task_token&);
198 
199   // Whether this is a blocker token.
200   bool is_blocker_;
201   // The number of blockers.
202   int blockers_;
203   // The single writer.
204   const Task* writer_;
205   // The list of Tasks waiting for this token to be released.
206   Task_list waiting_;
207 };
208 
209 // In order to support tokens more reliably, we provide objects which
210 // handle them using RAII.
211 
212 // RAII class to get a write lock on a token.  This requires
213 // specifying the task which is doing the lock.
214 
215 class Task_write_token
216 {
217  public:
Task_write_token(Task_token * token,const Task * task)218   Task_write_token(Task_token* token, const Task* task)
219     : token_(token), task_(task)
220   { this->token_->add_writer(this->task_); }
221 
~Task_write_token()222   ~Task_write_token()
223   { this->token_->remove_writer(this->task_); }
224 
225  private:
226   Task_write_token(const Task_write_token&);
227   Task_write_token& operator=(const Task_write_token&);
228 
229   Task_token* token_;
230   const Task* task_;
231 };
232 
233 // RAII class for a blocker.
234 
235 class Task_block_token
236 {
237  public:
238   // The blocker count must be incremented when the task is created.
239   // This object is created when the task is run, so we don't do
240   // anything in the constructor.
Task_block_token(Task_token * token)241   Task_block_token(Task_token* token)
242     : token_(token)
243   { gold_assert(this->token_->is_blocked()); }
244 
~Task_block_token()245   ~Task_block_token()
246   { this->token_->remove_blocker(); }
247 
248  private:
249   Task_block_token(const Task_block_token&);
250   Task_block_token& operator=(const Task_block_token&);
251 
252   Task_token* token_;
253 };
254 
255 // An object which implements an RAII lock for any object which
256 // supports lock and unlock methods.
257 
258 template<typename Obj>
259 class Task_lock_obj
260 {
261  public:
Task_lock_obj(const Task * task,Obj * obj)262   Task_lock_obj(const Task* task, Obj* obj)
263     : task_(task), obj_(obj)
264   { this->obj_->lock(task); }
265 
~Task_lock_obj()266   ~Task_lock_obj()
267   { this->obj_->unlock(this->task_); }
268 
269  private:
270   Task_lock_obj(const Task_lock_obj&);
271   Task_lock_obj& operator=(const Task_lock_obj&);
272 
273   const Task* task_;
274   Obj* obj_;
275 };
276 
277 // A class which holds the set of Task_tokens which must be locked for
278 // a Task.  No Task requires more than four Task_tokens, so we set
279 // that as a limit.
280 
281 class Task_locker
282 {
283  public:
284   static const int max_task_count = 4;
285 
Task_locker()286   Task_locker()
287     : count_(0)
288   { }
289 
~Task_locker()290   ~Task_locker()
291   { }
292 
293   // Clear the locker.
294   void
clear()295   clear()
296   { this->count_ = 0; }
297 
298   // Add a token to the locker.
299   void
add(Task * t,Task_token * token)300   add(Task* t, Task_token* token)
301   {
302     gold_assert(this->count_ < max_task_count);
303     this->tokens_[this->count_] = token;
304     ++this->count_;
305     // A blocker will have been incremented when the task is created.
306     // A writer we need to lock now.
307     if (!token->is_blocker())
308       token->add_writer(t);
309   }
310 
311   // Iterate over the tokens.
312 
313   typedef Task_token** iterator;
314 
315   iterator
begin()316   begin()
317   { return &this->tokens_[0]; }
318 
319   iterator
end()320   end()
321   { return &this->tokens_[this->count_]; }
322 
323  private:
324   Task_locker(const Task_locker&);
325   Task_locker& operator=(const Task_locker&);
326 
327   // The number of tokens.
328   int count_;
329   // The tokens.
330   Task_token* tokens_[max_task_count];
331 };
332 
333 } // End namespace gold.
334 
335 #endif // !defined(GOLD_TOKEN_H)
336