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