0
|
1 #include "SchedTaskArray.h"
|
|
2 #include "SchedTaskArrayNop.h"
|
|
3 #include "Scheduler.h"
|
|
4 #include "TaskManagerImpl.h"
|
|
5
|
|
6
|
|
7 SchedTaskArray::SchedTaskArray(Scheduler *s, SchedTaskBase *savedTask_, Task *curTask_, Task *_array, int tag)
|
|
8 {
|
|
9 savedTask = savedTask_;
|
|
10 atask = curTask_;
|
|
11 array = _array;
|
|
12 scheduler = s;
|
|
13
|
|
14 inListData.bound = 0;
|
|
15 inListData.size = 0;
|
|
16 inListData.length = 0;
|
|
17 inListData.element = 0;
|
|
18 outListData.bound = 0;
|
|
19 outListData.size = 0;
|
|
20 outListData.length = 0;
|
|
21 outListData.element = 0;
|
|
22
|
|
23 cur_index = -1;
|
|
24 task = 0;
|
|
25
|
|
26 this->tag = tag;
|
|
27
|
|
28 }
|
|
29
|
|
30 /**
|
|
31 Constructor for old Task with ListData
|
|
32 next TaskList entry contains Task object.
|
|
33 savedTask->rbuf is 0, it has only one Task.
|
|
34 */
|
|
35
|
|
36 SchedTaskArray::SchedTaskArray(Scheduler *s, SchedTaskBase *savedTask_)
|
|
37 {
|
|
38 savedTask = savedTask_;
|
|
39 scheduler = s;
|
|
40
|
|
41 inListData.bound = 0;
|
|
42 inListData.size = 0;
|
|
43 inListData.length = 0;
|
|
44 inListData.element = 0;
|
|
45 outListData.bound = 0;
|
|
46 outListData.size = 0;
|
|
47 outListData.length = 0;
|
|
48 outListData.element = 0;
|
|
49
|
|
50 SimpleTaskPtr st = &savedTask->list->tasks[savedTask->cur_index];
|
|
51 atask = (TaskPtr)st;
|
|
52 array = 0;
|
|
53 savedTask->cur_index += (atask->size()+sizeof(SimpleTask))/sizeof(SimpleTask);
|
|
54 cur_index = -1;
|
|
55 task = 0;
|
|
56
|
|
57 this->tag = 0;
|
|
58
|
|
59 }
|
|
60
|
|
61 /**
|
|
62 */
|
|
63 SchedTaskArray::~SchedTaskArray()
|
|
64 {
|
|
65 }
|
|
66
|
|
67 /**
|
|
68 * DMA buffer offset in rbuf
|
|
69 */
|
|
70 static void
|
|
71 bound(ListData *list)
|
|
72 {
|
|
73 ListElement *elm = list->element;
|
|
74 int *bound = list->bound;
|
|
75 int offset=0;
|
|
76 for(int i=0;i<list->length;i++) {
|
|
77 bound[i] = offset;
|
|
78 offset += elm[i].size;
|
|
79 }
|
|
80 }
|
|
81
|
|
82
|
|
83 /**
|
|
84 * Task data / code read
|
|
85 */
|
|
86 void
|
|
87 SchedTaskArray::read()
|
|
88 {
|
|
89
|
|
90 // object creation をSchedTaskArray生成時にやらないので、
|
|
91 // exec の直前のread で十分に間に合う
|
|
92 loadSchedTask(scheduler, atask->command);
|
|
93
|
|
94 // 読むデータが一つもなければ無視
|
|
95 if (atask->inData_count == 0) return;
|
|
96
|
|
97 inListData.length = atask->inData_count;
|
|
98 inListData.size = atask->inData_total_size();
|
|
99 inListData.element = atask->inData(0);
|
|
100 inListData.bound = (int*)manager->allocate(inListData.length*sizeof(int));
|
|
101
|
|
102 // load Input Data
|
|
103 readbuf = manager->allocate(inListData.size);
|
|
104 // inListData.print();
|
|
105 scheduler->dma_loadList(&inListData, readbuf, (DMA_READ + this->tag));
|
|
106 bound(&inListData);
|
|
107
|
|
108 }
|
|
109
|
|
110 /**
|
|
111 * Wait read data and execute task
|
|
112 * Start write DMA
|
|
113 */
|
|
114 void
|
|
115 SchedTaskArray::exec()
|
|
116 {
|
|
117 task_list[atask->command].wait(scheduler,atask->command);
|
|
118 TaskObjectRun run = task_list[atask->command].run;
|
|
119 if (atask->outData_count > 0) {
|
|
120 // allocate write buffer
|
|
121 outListData.length = atask->outData_count;
|
|
122 outListData.size = atask->outData_total_size();
|
|
123 // atask->outData_offset += cur_index + 1 ; // to avoid compiler bug
|
|
124 outListData.element = atask->outData(0);
|
|
125 outListData.bound = (int*)manager->allocate(outListData.length*sizeof(int));
|
|
126 bound(&outListData);
|
|
127
|
|
128 writebuf = manager->allocate(outListData.size);
|
|
129 //if (outListData.element == inListData.element ) {
|
|
130 // printf("bad %x\n",outListData.element);
|
|
131 //}
|
|
132 }
|
|
133 scheduler->dma_wait((DMA_READ + this->tag));
|
|
134 run(this, get_input(readbuf, 0), get_output(writebuf, 0));
|
|
135 free(readbuf);
|
|
136 // 書き込む領域がなければ無視
|
|
137
|
|
138 // User 側で作る方法が必要...
|
|
139
|
|
140 if (atask->outData_count > 0) {
|
|
141 // outListData.print();
|
|
142 scheduler->dma_storeList(&outListData, writebuf, DMA_WRITE);
|
|
143 }
|
|
144 }
|
|
145
|
|
146 /**
|
|
147 * Wait write DMA
|
|
148 * send finish mail
|
|
149 */
|
|
150 void
|
|
151 SchedTaskArray::write()
|
|
152 {
|
|
153
|
|
154 scheduler->dma_wait(DMA_WRITE);
|
|
155 free(writebuf);
|
|
156 free(inListData.bound);
|
|
157 free(outListData.bound);
|
|
158 }
|
|
159
|
|
160 Task *SchedTaskArray::last()
|
|
161 {
|
|
162 SchedTask *s = (SchedTask *)savedTask;
|
|
163 return (Task*)(((char*)array)+ s->read_size());
|
|
164 }
|
|
165
|
|
166 SchedTaskBase*
|
|
167 SchedTaskArray::next(Scheduler *scheduler, SchedTaskBase *p)
|
|
168 {
|
|
169
|
|
170 Task *next = atask->next();
|
|
171 if (next < last()) {
|
|
172 // Task List が残っているので、次を準備
|
|
173 //scheduler->printf("hog\n");
|
|
174 return new SchedTaskArray(scheduler, savedTask, next, array, this->tag^1);
|
|
175 } else {
|
|
176
|
|
177 //このTaskArrayは終わったが、Pipeline 上にread の TaskArray が残っているので
|
|
178 //1ステージを稼ぐ必要がある
|
|
179 //scheduler->printf("auau\n");
|
|
180 return new SchedTaskArrayNop(scheduler, savedTask, next, array);
|
|
181
|
|
182 }
|
|
183 }
|
|
184
|
|
185
|
|
186
|
|
187 /**
|
|
188 * task->add_inData で与えられた順番に対応する index (0〜n-1) で、
|
|
189 * buffer から対応するデータを返す。
|
|
190 */
|
|
191 void*
|
|
192 SchedTaskArray::get_input(void *buff, int index)
|
|
193 {
|
|
194 return (void*)((char*)readbuf + inListData.bound[index]);
|
|
195 }
|
|
196
|
|
197 /**
|
|
198 * get_input(index) のアドレスを返す
|
|
199 */
|
|
200 memaddr
|
|
201 SchedTaskArray::get_inputAddr(int index)
|
|
202 {
|
|
203 #ifdef __CERIUM_CELL__
|
|
204 return (memaddr)inListData.element[index].addr;
|
|
205 #else
|
|
206 return inListData.element[index].addr;
|
|
207 #endif
|
|
208 }
|
|
209
|
|
210 /**
|
|
211 * get_input(index) のサイズを返す
|
|
212 */
|
|
213 int
|
|
214 SchedTaskArray::get_inputSize(int index)
|
|
215 {
|
|
216 return inListData.element[index].size;
|
|
217 }
|
|
218
|
|
219 /**
|
|
220 * write buffer の領域を返す。
|
|
221 */
|
|
222 void*
|
|
223 SchedTaskArray::get_output(void *buff, int index)
|
|
224 {
|
|
225 return (void*)((char *)writebuf + outListData.bound[index]);
|
|
226 }
|
|
227
|
|
228 /**
|
|
229 * get_output(index) のアドレスを返す
|
|
230 */
|
|
231 memaddr
|
|
232 SchedTaskArray::get_outputAddr(int index)
|
|
233 {
|
|
234 #ifdef __CERIUM_CELL__
|
|
235 return (memaddr)outListData.element[index].addr;
|
|
236 #else
|
|
237 return outListData.element[index].addr;
|
|
238 #endif
|
|
239 }
|
|
240
|
|
241 /**
|
|
242 * get_output(index) のサイズを返す
|
|
243 */
|
|
244 int
|
|
245 SchedTaskArray::get_outputSize(int index)
|
|
246 {
|
|
247 return outListData.element[index].size;
|
|
248 }
|
|
249
|
|
250 memaddr
|
|
251 SchedTaskArray::get_param(int index)
|
|
252 {
|
|
253 return *atask->param(index);
|
|
254 }
|
|
255
|
|
256 int
|
|
257 SchedTaskArray::read_size()
|
|
258 {
|
|
259 return get_inputSize(0);
|
|
260 }
|
|
261
|
|
262
|
|
263 /* end */
|