• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // workqueue.h -- the work queue 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 // After processing the command line, everything the linker does is
24 // driven from a work queue.  This permits us to parallelize the
25 // linker where possible.
26 
27 #ifndef GOLD_WORKQUEUE_H
28 #define GOLD_WORKQUEUE_H
29 
30 #include <string>
31 
32 #include "gold-threads.h"
33 #include "token.h"
34 
35 namespace gold
36 {
37 
38 class General_options;
39 class Workqueue;
40 
41 // The superclass for tasks to be placed on the workqueue.  Each
42 // specific task class will inherit from this one.
43 
44 class Task
45 {
46  public:
Task()47   Task()
48     : list_next_(NULL), name_(), should_run_soon_(false)
49   { }
~Task()50   virtual ~Task()
51   { }
52 
53   // Check whether the Task can be run now.  This method is only
54   // called with the workqueue lock held.  If the Task can run, this
55   // returns NULL.  Otherwise it returns a pointer to a token which
56   // must be released before the Task can run.
57   virtual Task_token*
58   is_runnable() = 0;
59 
60   // Lock all the resources required by the Task, and store the locks
61   // in a Task_locker.  This method does not need to do anything if no
62   // locks are required.  This method is only called with the
63   // workqueue lock held.
64   virtual void
65   locks(Task_locker*) = 0;
66 
67   // Run the task.
68   virtual void
69   run(Workqueue*) = 0;
70 
71   // Return whether this task should run soon.
72   bool
should_run_soon()73   should_run_soon() const
74   { return this->should_run_soon_; }
75 
76   // Note that this task should run soon.
77   void
set_should_run_soon()78   set_should_run_soon()
79   { this->should_run_soon_ = true; }
80 
81   // Get the next Task on the list of Tasks.  Called by Task_list.
82   Task*
list_next()83   list_next() const
84   { return this->list_next_; }
85 
86   // Set the next Task on the list of Tasks.  Called by Task_list.
87   void
set_list_next(Task * t)88   set_list_next(Task* t)
89   {
90     gold_assert(this->list_next_ == NULL);
91     this->list_next_ = t;
92   }
93 
94   // Clear the next Task on the list of Tasks.  Called by Task_list.
95   void
clear_list_next()96   clear_list_next()
97   { this->list_next_ = NULL; }
98 
99   // Return the name of the Task.  This is only used for debugging
100   // purposes.
101   const std::string&
name()102   name()
103   {
104     if (this->name_.empty())
105       this->name_ = this->get_name();
106     return this->name_;
107   }
108 
109  protected:
110   // Get the name of the task.  This must be implemented by the child
111   // class.
112   virtual std::string
113   get_name() const = 0;
114 
115  private:
116   // Tasks may not be copied.
117   Task(const Task&);
118   Task& operator=(const Task&);
119 
120   // If this Task is on a list, this is a pointer to the next Task on
121   // the list.  We use this simple list structure rather than building
122   // a container, in order to avoid memory allocation while holding
123   // the Workqueue lock.
124   Task* list_next_;
125   // Task name, for debugging purposes.
126   std::string name_;
127   // Whether this Task should be executed soon.  This is used for
128   // Tasks which can be run after some data is read.
129   bool should_run_soon_;
130 };
131 
132 // An interface for Task_function.  This is a convenience class to run
133 // a single function.
134 
135 class Task_function_runner
136 {
137  public:
~Task_function_runner()138   virtual ~Task_function_runner()
139   { }
140 
141   virtual void
142   run(Workqueue*, const Task*) = 0;
143 };
144 
145 // A simple task which waits for a blocker and then runs a function.
146 
147 class Task_function : public Task
148 {
149  public:
150   // RUNNER and BLOCKER should be allocated using new, and will be
151   // deleted after the task runs.
Task_function(Task_function_runner * runner,Task_token * blocker,const char * name)152   Task_function(Task_function_runner* runner, Task_token* blocker,
153 		const char* name)
154     : runner_(runner), blocker_(blocker), name_(name)
155   { gold_assert(blocker != NULL); }
156 
~Task_function()157   ~Task_function()
158   {
159     delete this->runner_;
160     delete this->blocker_;
161   }
162 
163   // The standard task methods.
164 
165   // Wait until the task is unblocked.
166   Task_token*
is_runnable()167   is_runnable()
168   { return this->blocker_->is_blocked() ? this->blocker_ : NULL; }
169 
170   // This type of task does not normally hold any locks.
171   virtual void
locks(Task_locker *)172   locks(Task_locker*)
173   { }
174 
175   // Run the action.
176   void
run(Workqueue * workqueue)177   run(Workqueue* workqueue)
178   { this->runner_->run(workqueue, this); }
179 
180   // The debugging name.
181   std::string
get_name()182   get_name() const
183   { return this->name_; }
184 
185  private:
186   Task_function(const Task_function&);
187   Task_function& operator=(const Task_function&);
188 
189   Task_function_runner* runner_;
190   Task_token* blocker_;
191   const char* name_;
192 };
193 
194 // The workqueue itself.
195 
196 class Workqueue_threader;
197 
198 class Workqueue
199 {
200  public:
201   Workqueue(const General_options&);
202   ~Workqueue();
203 
204   // Add a new task to the work queue.
205   void
206   queue(Task*);
207 
208   // Add a new task to the work queue which should run soon.  If the
209   // task is ready, it will be run before any tasks added using
210   // queue().
211   void
212   queue_soon(Task*);
213 
214   // Add a new task to the work queue which should run next if it is
215   // ready.
216   void
217   queue_next(Task*);
218 
219   // Process all the tasks on the work queue.  This function runs
220   // until all tasks have completed.  The argument is the thread
221   // number, used only for debugging.
222   void
223   process(int);
224 
225   // Set the desired thread count--the number of threads we want to
226   // have running.
227   void
228   set_thread_count(int);
229 
230   // Add a new blocker to an existing Task_token. This must be done
231   // with the workqueue lock held.  This should not be done routinely,
232   // only in special circumstances.
233   void
234   add_blocker(Task_token*);
235 
236  private:
237   // This class can not be copied.
238   Workqueue(const Workqueue&);
239   Workqueue& operator=(const Workqueue&);
240 
241   // Add a task to a queue.
242   void
243   add_to_queue(Task_list* queue, Task* t, bool front);
244 
245   // Find a runnable task, or wait for one.
246   Task*
247   find_runnable_or_wait(int thread_number);
248 
249   // Find a runnable task.
250   Task*
251   find_runnable();
252 
253   // Find a runnable task in a list.
254   Task*
255   find_runnable_in_list(Task_list*);
256 
257   // Find an run a task.
258   bool
259   find_and_run_task(int);
260 
261   // Release the locks for a Task.  Return the next Task to run.
262   Task*
263   release_locks(Task*, Task_locker*);
264 
265   // Store T into *PRET, or queue it as appropriate.
266   bool
267   return_or_queue(Task* t, bool is_blocker, Task** pret);
268 
269   // Return whether to cancel this thread.
270   bool
271   should_cancel_thread(int thread_number);
272 
273   // Master Workqueue lock.  This controls access to the following
274   // member variables.
275   Lock lock_;
276   // List of tasks to execute soon.
277   Task_list first_tasks_;
278   // List of tasks to execute after the ones in first_tasks_.
279   Task_list tasks_;
280   // Number of tasks currently running.
281   int running_;
282   // Number of tasks waiting for a lock to release.
283   int waiting_;
284   // Condition variable associated with lock_.  This is signalled when
285   // there may be a new Task to execute.
286   Condvar condvar_;
287 
288   // The threading implementation.  This is set at construction time
289   // and not changed thereafter.
290   Workqueue_threader* threader_;
291 };
292 
293 } // End namespace gold.
294 
295 #endif // !defined(GOLD_WORKQUEUE_H)
296