Annotation of sys/dev/raidframe/rf_cvscan.c, Revision 1.1.1.1
1.1 nbrk 1: /* $OpenBSD: rf_cvscan.c,v 1.5 2002/12/16 07:01:03 tdeval Exp $ */
2: /* $NetBSD: rf_cvscan.c,v 1.5 1999/08/13 03:41:53 oster Exp $ */
3:
4: /*
5: * Copyright (c) 1995 Carnegie-Mellon University.
6: * All rights reserved.
7: *
8: * Author: Mark Holland
9: *
10: * Permission to use, copy, modify and distribute this software and
11: * its documentation is hereby granted, provided that both the copyright
12: * notice and this permission notice appear in all copies of the
13: * software, derivative works or modified versions, and any portions
14: * thereof, and that both notices appear in supporting documentation.
15: *
16: * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
17: * CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND
18: * FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
19: *
20: * Carnegie Mellon requests users of this software to return to
21: *
22: * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
23: * School of Computer Science
24: * Carnegie Mellon University
25: * Pittsburgh PA 15213-3890
26: *
27: * any improvements or extensions that they make and grant Carnegie the
28: * rights to redistribute these changes.
29: */
30:
31: /*****************************************************************************
32: *
33: * cvscan.c -- prioritized cvscan disk queueing code.
34: *
35: * Nov 9, 1994, adapted from raidSim version (MCH)
36: *
37: *****************************************************************************/
38:
39: #include "rf_types.h"
40: #include "rf_alloclist.h"
41: #include "rf_stripelocks.h"
42: #include "rf_layout.h"
43: #include "rf_diskqueue.h"
44: #include "rf_cvscan.h"
45: #include "rf_debugMem.h"
46: #include "rf_general.h"
47:
48: void rf_CheckCvscanState(RF_CvscanHeader_t *, char *, int);
49: void rf_PriorityInsert(RF_DiskQueueData_t **, RF_DiskQueueData_t *);
50: void rf_ReqInsert(RF_DiskQueueData_t **, RF_DiskQueueData_t *,
51: RF_CvscanArmDir_t);
52: RF_DiskQueueData_t *rf_ReqDequeue(RF_DiskQueueData_t **);
53: void rf_ReBalance(RF_CvscanHeader_t *);
54: void rf_Transfer(RF_DiskQueueData_t **, RF_DiskQueueData_t **);
55: void rf_RealEnqueue(RF_CvscanHeader_t *, RF_DiskQueueData_t *);
56:
57: #define DO_CHECK_STATE(_hdr_) rf_CheckCvscanState((_hdr_), __FILE__, __LINE__)
58:
59: #define pri_ok(p) (((p) == RF_IO_NORMAL_PRIORITY) || \
60: ((p) == RF_IO_LOW_PRIORITY))
61:
62:
63: void
64: rf_CheckCvscanState(RF_CvscanHeader_t *hdr, char *file, int line)
65: {
66: long i, key;
67: RF_DiskQueueData_t *tmp;
68:
69: if (hdr->left != (RF_DiskQueueData_t *) NULL)
70: RF_ASSERT(hdr->left->sectorOffset < hdr->cur_block);
71: for (key = hdr->cur_block, i = 0, tmp = hdr->left;
72: tmp != (RF_DiskQueueData_t *) NULL;
73: key = tmp->sectorOffset, i++, tmp = tmp->next)
74: RF_ASSERT(tmp->sectorOffset <= key
75: && tmp->priority == hdr->nxt_priority &&
76: pri_ok(tmp->priority));
77: RF_ASSERT(i == hdr->left_cnt);
78:
79: for (key = hdr->cur_block, i = 0, tmp = hdr->right;
80: tmp != (RF_DiskQueueData_t *) NULL;
81: key = tmp->sectorOffset, i++, tmp = tmp->next) {
82: RF_ASSERT(key <= tmp->sectorOffset);
83: RF_ASSERT(tmp->priority == hdr->nxt_priority);
84: RF_ASSERT(pri_ok(tmp->priority));
85: }
86: RF_ASSERT(i == hdr->right_cnt);
87:
88: for (key = hdr->nxt_priority - 1, tmp = hdr->burner;
89: tmp != (RF_DiskQueueData_t *) NULL;
90: key = tmp->priority, tmp = tmp->next) {
91: RF_ASSERT(tmp);
92: RF_ASSERT(hdr);
93: RF_ASSERT(pri_ok(tmp->priority));
94: RF_ASSERT(key >= tmp->priority);
95: RF_ASSERT(tmp->priority < hdr->nxt_priority);
96: }
97: }
98:
99:
100: void
101: rf_PriorityInsert(RF_DiskQueueData_t **list_ptr, RF_DiskQueueData_t *req)
102: {
103: /*
104: * Insert block pointed to by req into list whose first entry is
105: * pointed to by the pointer that list_ptr points to.
106: * i.e. list_ptr is a grandparent of the first entry.
107: */
108:
109: for (; (*list_ptr) != (RF_DiskQueueData_t *) NULL &&
110: (*list_ptr)->priority > req->priority;
111: list_ptr = &((*list_ptr)->next)) {
112: }
113: req->next = (*list_ptr);
114: (*list_ptr) = req;
115: }
116:
117:
118: void
119: rf_ReqInsert(RF_DiskQueueData_t **list_ptr, RF_DiskQueueData_t *req,
120: RF_CvscanArmDir_t order)
121: {
122: /*
123: * Insert block pointed to by req into list whose first entry is
124: * pointed to by the pointer that list_ptr points to.
125: * i.e. list_ptr is a grandparent of the first entry.
126: */
127:
128: for (; (*list_ptr) != (RF_DiskQueueData_t *) NULL &&
129: ((order == rf_cvscan_RIGHT && (*list_ptr)->sectorOffset <=
130: req->sectorOffset) || (order == rf_cvscan_LEFT &&
131: (*list_ptr)->sectorOffset > req->sectorOffset));
132: list_ptr = &((*list_ptr)->next)) {
133: }
134: req->next = (*list_ptr);
135: (*list_ptr) = req;
136: }
137:
138:
139: RF_DiskQueueData_t *
140: rf_ReqDequeue(RF_DiskQueueData_t **list_ptr)
141: {
142: RF_DiskQueueData_t *ret = (*list_ptr);
143: if ((*list_ptr) != (RF_DiskQueueData_t *) NULL) {
144: (*list_ptr) = (*list_ptr)->next;
145: }
146: return (ret);
147: }
148:
149:
150: void
151: rf_ReBalance(RF_CvscanHeader_t *hdr)
152: {
153: /* DO_CHECK_STATE(hdr); */
154: while (hdr->right != (RF_DiskQueueData_t *) NULL
155: && hdr->right->sectorOffset < hdr->cur_block) {
156: hdr->right_cnt--;
157: hdr->left_cnt++;
158: rf_ReqInsert(&hdr->left, rf_ReqDequeue(&hdr->right),
159: rf_cvscan_LEFT);
160: }
161: /* DO_CHECK_STATE(hdr); */
162: }
163:
164:
165: void
166: rf_Transfer(RF_DiskQueueData_t **to_list_ptr, RF_DiskQueueData_t **from_list_ptr)
167: {
168: RF_DiskQueueData_t *gp;
169: for (gp = (*from_list_ptr); gp != (RF_DiskQueueData_t *) NULL;) {
170: RF_DiskQueueData_t *p = gp->next;
171: rf_PriorityInsert(to_list_ptr, gp);
172: gp = p;
173: }
174: (*from_list_ptr) = (RF_DiskQueueData_t *) NULL;
175: }
176:
177:
178: void
179: rf_RealEnqueue(RF_CvscanHeader_t *hdr, RF_DiskQueueData_t *req)
180: {
181: RF_ASSERT(req->priority == RF_IO_NORMAL_PRIORITY ||
182: req->priority == RF_IO_LOW_PRIORITY);
183:
184: DO_CHECK_STATE(hdr);
185: if (hdr->left_cnt == 0 && hdr->right_cnt == 0) {
186: hdr->nxt_priority = req->priority;
187: }
188: if (req->priority > hdr->nxt_priority) {
189: /*
190: * Dump all other outstanding requests on the back burner.
191: */
192: rf_Transfer(&hdr->burner, &hdr->left);
193: rf_Transfer(&hdr->burner, &hdr->right);
194: hdr->left_cnt = 0;
195: hdr->right_cnt = 0;
196: hdr->nxt_priority = req->priority;
197: }
198: if (req->priority < hdr->nxt_priority) {
199: /*
200: * Yet another low priority task !
201: */
202: rf_PriorityInsert(&hdr->burner, req);
203: } else {
204: if (req->sectorOffset < hdr->cur_block) {
205: /* This request is to the left of the current arms. */
206: rf_ReqInsert(&hdr->left, req, rf_cvscan_LEFT);
207: hdr->left_cnt++;
208: } else {
209: /* This request is to the right of the current arms. */
210: rf_ReqInsert(&hdr->right, req, rf_cvscan_RIGHT);
211: hdr->right_cnt++;
212: }
213: }
214: DO_CHECK_STATE(hdr);
215: }
216:
217:
218: void
219: rf_CvscanEnqueue(void *q_in, RF_DiskQueueData_t *elem, int priority)
220: {
221: RF_CvscanHeader_t *hdr = (RF_CvscanHeader_t *) q_in;
222: rf_RealEnqueue(hdr, elem /* req */ );
223: }
224:
225:
226: RF_DiskQueueData_t *
227: rf_CvscanDequeue(void *q_in)
228: {
229: RF_CvscanHeader_t *hdr = (RF_CvscanHeader_t *) q_in;
230: long range, i, sum_dist_left, sum_dist_right;
231: RF_DiskQueueData_t *ret;
232: RF_DiskQueueData_t *tmp;
233:
234: DO_CHECK_STATE(hdr);
235:
236: if (hdr->left_cnt == 0 && hdr->right_cnt == 0)
237: return ((RF_DiskQueueData_t *) NULL);
238:
239: range = RF_MIN(hdr->range_for_avg, RF_MIN(hdr->left_cnt,
240: hdr->right_cnt));
241: for (i = 0, tmp = hdr->left, sum_dist_left =
242: ((hdr->direction == rf_cvscan_RIGHT) ?
243: range * hdr->change_penalty : 0);
244: tmp != (RF_DiskQueueData_t *) NULL && i < range;
245: tmp = tmp->next, i++) {
246: sum_dist_left += hdr->cur_block - tmp->sectorOffset;
247: }
248: for (i = 0, tmp = hdr->right, sum_dist_right =
249: ((hdr->direction == rf_cvscan_LEFT) ?
250: range * hdr->change_penalty : 0);
251: tmp != (RF_DiskQueueData_t *) NULL && i < range;
252: tmp = tmp->next, i++) {
253: sum_dist_right += tmp->sectorOffset - hdr->cur_block;
254: }
255:
256: if (hdr->right_cnt == 0 || sum_dist_left < sum_dist_right) {
257: hdr->direction = rf_cvscan_LEFT;
258: hdr->cur_block = hdr->left->sectorOffset + hdr->left->numSector;
259: hdr->left_cnt = RF_MAX(hdr->left_cnt - 1, 0);
260: tmp = hdr->left;
261: ret = (rf_ReqDequeue(&hdr->left)) /*->parent*/ ;
262: } else {
263: hdr->direction = rf_cvscan_RIGHT;
264: hdr->cur_block = hdr->right->sectorOffset +
265: hdr->right->numSector;
266: hdr->right_cnt = RF_MAX(hdr->right_cnt - 1, 0);
267: tmp = hdr->right;
268: ret = (rf_ReqDequeue(&hdr->right)) /*->parent*/ ;
269: }
270: rf_ReBalance(hdr);
271:
272: if (hdr->left_cnt == 0 && hdr->right_cnt == 0
273: && hdr->burner != (RF_DiskQueueData_t *) NULL) {
274: /*
275: * Restore low priority requests for next dequeue.
276: */
277: RF_DiskQueueData_t *burner = hdr->burner;
278: hdr->nxt_priority = burner->priority;
279: while (burner != (RF_DiskQueueData_t *) NULL &&
280: burner->priority == hdr->nxt_priority) {
281: RF_DiskQueueData_t *next = burner->next;
282: rf_RealEnqueue(hdr, burner);
283: burner = next;
284: }
285: hdr->burner = burner;
286: }
287: DO_CHECK_STATE(hdr);
288: return (ret);
289: }
290:
291:
292: RF_DiskQueueData_t *
293: rf_CvscanPeek(void *q_in)
294: {
295: RF_CvscanHeader_t *hdr = (RF_CvscanHeader_t *) q_in;
296: long range, i, sum_dist_left, sum_dist_right;
297: RF_DiskQueueData_t *tmp, *headElement;
298:
299: DO_CHECK_STATE(hdr);
300:
301: if (hdr->left_cnt == 0 && hdr->right_cnt == 0)
302: headElement = NULL;
303: else {
304: range = RF_MIN(hdr->range_for_avg, RF_MIN(hdr->left_cnt,
305: hdr->right_cnt));
306: for (i = 0, tmp = hdr->left, sum_dist_left =
307: ((hdr->direction == rf_cvscan_RIGHT) ?
308: range * hdr->change_penalty : 0);
309: tmp != (RF_DiskQueueData_t *) NULL && i < range;
310: tmp = tmp->next, i++) {
311: sum_dist_left += hdr->cur_block - tmp->sectorOffset;
312: }
313: for (i = 0, tmp = hdr->right, sum_dist_right =
314: ((hdr->direction == rf_cvscan_LEFT) ?
315: range * hdr->change_penalty : 0);
316: tmp != (RF_DiskQueueData_t *) NULL && i < range;
317: tmp = tmp->next, i++) {
318: sum_dist_right += tmp->sectorOffset - hdr->cur_block;
319: }
320:
321: if (hdr->right_cnt == 0 || sum_dist_left < sum_dist_right)
322: headElement = hdr->left;
323: else
324: headElement = hdr->right;
325: }
326: return (headElement);
327: }
328:
329:
330: /*
331: * CVSCAN( 1, 0 ) is Shortest Seek Time First (SSTF)
332: * lowest average response time
333: * CVSCAN( 1, infinity ) is SCAN
334: * lowest response time standard deviation
335: */
336:
337:
338: int
339: rf_CvscanConfigure(void)
340: {
341: return (0);
342: }
343:
344:
345: void *
346: rf_CvscanCreate(RF_SectorCount_t sectPerDisk, RF_AllocListElem_t *clList,
347: RF_ShutdownList_t **listp)
348: {
349: RF_CvscanHeader_t *hdr;
350: long range = 2; /* Currently no mechanism to change these. */
351: long penalty = sectPerDisk / 5;
352:
353: RF_MallocAndAdd(hdr, sizeof(RF_CvscanHeader_t), (RF_CvscanHeader_t *),
354: clList);
355: bzero((char *) hdr, sizeof(RF_CvscanHeader_t));
356: hdr->range_for_avg = RF_MAX(range, 1);
357: hdr->change_penalty = RF_MAX(penalty, 0);
358: hdr->direction = rf_cvscan_RIGHT;
359: hdr->cur_block = 0;
360: hdr->left_cnt = hdr->right_cnt = 0;
361: hdr->left = hdr->right = (RF_DiskQueueData_t *) NULL;
362: hdr->burner = (RF_DiskQueueData_t *) NULL;
363: DO_CHECK_STATE(hdr);
364:
365: return ((void *) hdr);
366: }
367:
368:
369: #if (defined(__NetBSD__) || defined(__OpenBSD__)) && defined(_KERNEL)
370: /* rf_PrintCvscanQueue is not used, so we ignore it... */
371: #else
372: void
373: rf_PrintCvscanQueue(RF_CvscanHeader_t *hdr)
374: {
375: RF_DiskQueueData_t *tmp;
376:
377: printf("CVSCAN(%d,%d) at %d going %s\n",
378: (int) hdr->range_for_avg,
379: (int) hdr->change_penalty,
380: (int) hdr->cur_block,
381: (hdr->direction == rf_cvscan_LEFT) ? "LEFT" : "RIGHT");
382: printf("\tLeft(%d): ", hdr->left_cnt);
383: for (tmp = hdr->left; tmp != (RF_DiskQueueData_t *) NULL;
384: tmp = tmp->next)
385: printf("(%d,%ld,%d) ",
386: (int) tmp->sectorOffset,
387: (long) (tmp->sectorOffset + tmp->numSector),
388: tmp->priority);
389: printf("\n");
390: printf("\tRight(%d): ", hdr->right_cnt);
391: for (tmp = hdr->right; tmp != (RF_DiskQueueData_t *) NULL;
392: tmp = tmp->next)
393: printf("(%d,%ld,%d) ",
394: (int) tmp->sectorOffset,
395: (long) (tmp->sectorOffset + tmp->numSector),
396: tmp->priority);
397: printf("\n");
398: printf("\tBurner: ");
399: for (tmp = hdr->burner; tmp != (RF_DiskQueueData_t *) NULL;
400: tmp = tmp->next)
401: printf("(%d,%ld,%d) ",
402: (int) tmp->sectorOffset,
403: (long) (tmp->sectorOffset + tmp->numSector),
404: tmp->priority);
405: printf("\n");
406: }
407: #endif
408:
409:
410: /*
411: * Promote reconstruction accesses for the given stripeID to normal priority.
412: * Return 1 if an access was found and zero otherwise.
413: * Normally, we should only have one or zero entries in the burner queue,
414: * so execution time should be short.
415: */
416: int
417: rf_CvscanPromote(void *q_in, RF_StripeNum_t parityStripeID,
418: RF_ReconUnitNum_t which_ru)
419: {
420: RF_CvscanHeader_t *hdr = (RF_CvscanHeader_t *) q_in;
421: RF_DiskQueueData_t *trailer = NULL, *tmp = hdr->burner, *tlist = NULL;
422: int retval = 0;
423:
424: DO_CHECK_STATE(hdr);
425: while (tmp) { /* Handle entries at the front of the list. */
426: if (tmp->parityStripeID == parityStripeID &&
427: tmp->which_ru == which_ru) {
428: hdr->burner = tmp->next;
429: tmp->priority = RF_IO_NORMAL_PRIORITY;
430: tmp->next = tlist;
431: tlist = tmp;
432: tmp = hdr->burner;
433: } else
434: break;
435: }
436: if (tmp) {
437: trailer = tmp;
438: tmp = tmp->next;
439: }
440: while (tmp) { /* Handle entries on the rest of the list. */
441: if (tmp->parityStripeID == parityStripeID &&
442: tmp->which_ru == which_ru) {
443: trailer->next = tmp->next;
444: tmp->priority = RF_IO_NORMAL_PRIORITY;
445: tmp->next = tlist;
446: tlist = tmp; /* Insert on a temp queue. */
447: tmp = trailer->next;
448: } else {
449: trailer = tmp;
450: tmp = tmp->next;
451: }
452: }
453: while (tlist) {
454: retval++;
455: tmp = tlist->next;
456: rf_RealEnqueue(hdr, tlist);
457: tlist = tmp;
458: }
459: RF_ASSERT(retval == 0 || retval == 1);
460: DO_CHECK_STATE((RF_CvscanHeader_t *) q_in);
461: return (retval);
462: }
CVSweb