contributed by < Pepitaw
>
linux2022
$ gcc --version
gcc (Ubuntu 9.3.0-17ubuntu1~20.04) 9.3.0
$ lscpu
Architecture: x86_64
CPU op-mode(s): 32-bit, 64-bit
Byte Order: Little Endian
Address sizes: 39 bits physical, 48 bits virtual
CPU(s): 12
On-line CPU(s) list: 0-11
Thread(s) per core: 2
Core(s) per socket: 6
Socket(s): 1
NUMA node(s): 1
Vendor ID: GenuineIntel
CPU family: 6
Model: 158
Model name: Intel(R) Core(TM) i7-9750H CPU @ 2.60GHz
Stepping: 10
CPU MHz: 2600.000
CPU max MHz: 4500.0000
CPU min MHz: 800.0000
BogoMIPS: 5199.98
L1d cache: 192 KiB
L1i cache: 192 KiB
L2 cache: 1.5 MiB
L3 cache: 12 MiB
NUMA node0 CPU(s): 0-11
作業要求
根據 list.h
參考函式 INIT_LIST_HEAD
初始化 q ,將 q 的 next
與 prev
指向自己。
struct list_head *q_new()
{
struct list_head *q = malloc(sizeof(struct list_head));
if (!q)
return NULL;
INIT_LIST_HEAD(q);
return q;
}
malloc
分配記憶體空間給指標 qmalloc
分配空間失敗,會回傳 NULL
INIT_LIST_HEAD
初始化指標 q根據 list.h
參考函式 list_for_each_entry_safe
,藉由副本遍歷各個 entry
,不會受到移除 entry
的影響
void q_free(struct list_head *l)
{
if (!l)
return;
element_t *pos, *n;
list_for_each_entry_safe (pos, n, l, list)
q_release_element(pos);
free(l);
}
head
指向 NULL 就 return
linked list
,對每個 entry
呼叫 q_release_element
釋放其空間head
的記憶體空間根據 list.h
參考函式 list_add
,將新增的 element_t
加到 head
的 next
bool q_insert_head(struct list_head *head, char *s)
{
if (!head)
return false;
element_t *e = malloc(sizeof(element_t));
if (!e)
return false;
e->value = strdup(s);
if (!e->value) {
free(e);
return false;
}
list_add(&e->list, head);
return true;
}
head
指向 NULL 就回傳 false
malloc()
分配 element_t
大小的記憶體空間給 e ,若分配失敗則回傳 false
strdup
會用 malloc
分配記憶體儲存字串的特性,直接指派給 e->valuestrdup
分配記憶體失敗就清除 e 與回傳 false
entry
到listbool q_insert_head(struct list_head *head, char *s)
{
struct node{
char *str;
struct list_head list;
};
struct node *n = malloc(sizeof(struct node));
strcpy(n->str, s);
list_add(&n->list, head);
return true;
}
element_t *e = malloc(sizeof(element_t));
if(!e)
return false;
strcpy(e->value, s);
list_add(&e->list, head);
return true;
struct list_head *tmp = head->next;
list_del(tmp);
list_add(tmp, head->next->next);
由以下結果得知,可以直接插入指定位置的下一個 entry
cmd> ih ant
l = [ant dolphin dolphin dolphin gerbil bear dolphin meerkat bear bear bear cat]
cmd> reverse
l = [dolphin dolphin ant dolphin gerbil bear dolphin meerkat bear bear bear cat]
根據 list.h
參考函式 list_add_tail
,將新增的 element_t
加到 head
的 prev
,其餘同 q_insert_head
bool q_insert_tail(struct list_head *head, char *s)
{
if (!head)
return false;
element_t *e = malloc(sizeof(element_t));
if (!e)
return false;
e->value = strdup(s);
if (!e->value) {
free(e);
return false;
}
list_add_tail(&e->list, head);
return true;
}
根據 list.h
參考函式 list_first_entry
,找到第一個 entry
,也參考函式 list_del
將目標 entry
remove ,成為單一節點
element_t *q_remove_head(struct list_head *head, char *sp, size_t bufsize)
{
if (!head || head->next == head)
return NULL;
element_t *e = list_first_entry(head, element_t, list);
list_del(&e->list);
if (sp) {
strncpy(sp, e->value, bufsize - 1);
sp[bufsize - 1] = '\0';
}
return e;
}
head
指向 NULL 或只有 head
就回傳 NULL
list_first_entry
找到第一個 entry
list_del
將目標 entry
remove ,成為單一節點\0
\0
,但卻報錯還不知道原因。if(sp){
sp = strndup(e->value, bufsize - 1);
}
ERROR: Failed to store removed value
根據 list.h
參考函式 list_last_entry
,找到最後一個 entry
,其餘與 q_remove_head
相同
element_t *q_remove_head(struct list_head *head, char *sp, size_t bufsize)
{
if (!head || head->next == head)
return NULL;
element_t *e = list_last_entry(head, element_t, list);
list_del(&e->list);
if (sp) {
strncpy(sp, e->value, bufsize - 1);
sp[bufsize - 1] = '\0';
}
return e;
}
根據 list.h
參考函式 list_for_each()
遍歷各個 entry
int q_size(struct list_head *head)
{
if (!head)
return false;
int i = 0;
struct list_head *lh;
list_for_each (lh, head)
i++;
return i;
}
head
指向 NULL 就回傳 false
entry
i 就加一,最後即為size依據「龜兔賽跑」Floyd’s Cycle detection,當 rabbit
跑完 list 時, turtle
跑到一半的特性,來尋找中間的節點
bool q_delete_mid(struct list_head *head)
{
struct list_head *fast = head->next, *slow = head->next;
for (; fast != head->prev && fast->next != head->prev;
fast = fast->next->next, slow = slow->next)
;
/*while(fast != NULL && fast->next != NULL){
fast = fast->next->next;
slow = slow->next;
}*/
slow->prev->next = slow->next;
slow->next->prev = slow->prev;
q_release_element(list_entry(slow, element_t, list));
return true;
}
entry
, slow 跑兩個 entry
for(struct list_head *fast = head, *slow = head; fast && fast->next; \
fast = fast->next->next, slow = slow->next);
bool q_delete_dup(struct list_head *head)
{
if (!head)
return false;
element_t *pos, *n;
bool used = false;
list_for_each_entry_safe (pos, n, head, list) {
if (pos->list.next != head &&
!strcmp(pos->value,
list_entry(pos->list.next, element_t, list)->value)) {
list_del(&pos->list);
q_release_element(pos);
used = true;
} else if (used) {
list_del(&pos->list);
q_release_element(pos);
used = false;
}
}
return true;
}
head
指向 NULL 就回傳 false
list_for_each_entry_safe
遍歷各個 entry
strcmp
比較 pos 跟下一個 entry
的 value 是否相同entry
是否跟以刪除 entry
有相同的 value根據 list.h
參考函式 list_move_tail
將 entry
放到 head->prev
void q_swap(struct list_head *head)
{
bool odd = q_size(head) & 0x01;
for (int i = q_size(head) >> 1; i > 0; i--) {
list_move_tail(head->next->next, head);
list_move_tail(head->next, head);
}
if (odd)
list_move_tail(head->next, head);
}
entry
放到尾巴entry
放到尾巴list.h
函式 list_swap
,但發現未定義void q_shuffle(struct list_head *head)
{
srand( time(NULL) );
for (int i = q_size(head); i > 0; i--) {
struct list_head *tmp = head->next;
for (int x = rand()%i; x > 0; x--){
tmp = tmp->next;
}
list_swap(tmp, head->prev);
//list_move_tail(tmp, head);
}
}
queue.c: In function ‘q_shuffle’:
queue.c:347:2: error: implicit declaration of function ‘list_swap’ [-Werror=implicit-function-declaration]
347 | list_swap(tmp, head->prev);
| ^~~~~~~~~
cc1: all warnings being treated as errors
make: *** [Makefile:50:queue.o] 錯誤 1
void q_reverse(struct list_head *head)
{
if (!head || head->next->next == head)
return;
struct list_head *pos = head->prev;
for (int i = q_size(head); i > 1; i--)
list_move_tail(pos->prev, head);
}
head
指向 NULL 或 list 只有一個 entry
就回傳 NULLentry
依序放到尾巴struct list_head *mergesort(struct list_head *node)
{
if (!node || !node->next)
return node;
struct list_head *fast = node, *slow = node;
for (; fast->next && fast->next->next;
fast = fast->next->next, slow = slow->next)
;
fast = slow->next;
slow->next = NULL;
struct list_head *l1 = mergesort(node);
struct list_head *l2 = mergesort(fast);
struct list_head t_head, *tmp = &t_head;
while (l1 && l2) {
if (strcmp(list_entry(l1, element_t, list)->value,
list_entry(l2, element_t, list)->value) > 0) {
tmp->next = l2;
l2 = l2->next;
} else {
tmp->next = l1;
l1 = l1->next;
}
tmp = tmp->next;
}
if (l1)
tmp->next = l1;
else
tmp->next = l2;
return t_head.next;
}
void q_sort(struct list_head *head)
{
if (!head || !head->next)
return;
head->prev->next = NULL;
struct list_head *list = head->next;
list = mergesort(list);
head->next = list;
struct list_head *i = head;
while (i->next != NULL) {
i->next->prev = i;
i = i->next;
}
head->prev = i;
i->next = head;
}
head
指向 NULL 或 list 沒有 entry
就 returnhead
的 list 進行 mergesortstrcmp
比較大小,依小到大排列head
指向 排序好的 listlist
的 next->prev
將所有 prev
重新排列static struct list_head *merge(void *priv, list_cmp_func_t cmp,
struct list_head *a, struct list_head *b)
{
struct list_head *head, **tail = &head;
for (;;) {
/* if equal, take 'a' -- important for sort stability */
if (cmp(priv, a, b) <= 0) {
*tail = a;
tail = &a->next;
a = a->next;
if (!a) {
*tail = b;
break;
}
} else {
*tail = b;
tail = &b->next;
b = b->next;
if (!b) {
*tail = a;
break;
}
}
}
return head;
}
static void merge_final(void *priv, list_cmp_func_t cmp, struct list_head *head,
struct list_head *a, struct list_head *b)
{
struct list_head *tail = head;
u8 count = 0;
for (;;) {
/* if equal, take 'a' -- important for sort stability */
if (cmp(priv, a, b) <= 0) {
tail->next = a;
a->prev = tail;
tail = a;
a = a->next;
if (!a)
break;
} else {
tail->next = b;
b->prev = tail;
tail = b;
b = b->next;
if (!b) {
b = a;
break;
}
}
}
/* Finish linking remainder of list b on to tail */
tail->next = b;
do {
/*
* If the merge is highly unbalanced (e.g. the input is
* already sorted), this loop may run many iterations.
* Continue callbacks to the client even though no
* element comparison is needed, so the client's cmp()
* routine can invoke cond_resched() periodically.
*/
if (unlikely(!++count))
cmp(priv, b, b);
b->prev = tail;
tail = b;
b = b->next;
} while (b);
/* And the final links to make a circular doubly-linked list */
tail->next = head;
head->prev = tail;
}
void list_sort(void *priv, struct list_head *head, list_cmp_func_t cmp)
{
struct list_head *list = head->next, *pending = NULL;
size_t count = 0; /* Count of pending */
if (list == head->prev) /* Zero or one elements */
return;
/* Convert to a null-terminated singly-linked list. */
head->prev->next = NULL;
do {
size_t bits;
struct list_head **tail = &pending;
/* Find the least-significant clear bit in count */
for (bits = count; bits & 1; bits >>= 1)
tail = &(*tail)->prev;
/* Do the indicated merge */
if (likely(bits)) {
struct list_head *a = *tail, *b = a->prev;
a = merge(priv, cmp, b, a);
/* Install the merged result in place of the inputs */
a->prev = b->prev;
*tail = a;
}
/* Move one element from input list to pending */
list->prev = pending;
pending = list;
list = list->next;
pending->next = NULL;
count++;
} while (list);
/* End of input; merge together all the pending lists. */
list = pending;
pending = pending->prev;
for (;;) {
struct list_head *next = pending->prev;
if (!next)
break;
list = merge(priv, cmp, pending, list);
pending = next;
}
/* The final merge, rebuilding prev links */
merge_final(priv, cmp, head, pending, list);
}
作業目標
更新中
$ valgrind -q --leak-check=full ./qtest -f traces/trace-01-ops.cmd
==2951== 5 bytes in 1 blocks are still reachable in loss record 1 of 2
==2951== at 0x483B7F3: malloc (in /usr/lib/x86_64-linux-gnu/valgrind/vgpreload_memcheck-amd64-linux.so)
==2951== by 0x4A5250E: strdup (strdup.c:42)
==2951== by 0x1108BE: linenoiseHistoryAdd (linenoise.c:1236)
==2951== by 0x111451: linenoiseHistoryLoad (linenoise.c:1325)
==2951== by 0x10C6B0: main (qtest.c:951)
==2951==
==2951== 160 bytes in 1 blocks are still reachable in loss record 2 of 2
==2951== at 0x483B7F3: malloc (in /usr/lib/x86_64-linux-gnu/valgrind/vgpreload_memcheck-amd64-linux.so)
==2951== by 0x11087E: linenoiseHistoryAdd (linenoise.c:1224)
==2951== by 0x111451: linenoiseHistoryLoad (linenoise.c:1325)
==2951== by 0x10C6B0: main (qtest.c:951)
linecopy = strdup(line);
作業目標
qtest
提供新的命令 shuffle
,允許藉由 Fisher–Yates shuffle 演算法,對佇列中所有節點進行洗牌 (shuffle) 操作void q_shuffle(struct list_head *head)
{
srand(time(NULL));
for (int i = q_size(head); i > 0; i--) {
struct list_head *tmp = head->next, *tail = head->prev;
for (int x = rand() % i; x > 0; x--)
tmp = tmp->next;
list_del(tail);
list_add(tail, tmp->prev);
list_move_tail(tmp, head);
}
}
linked list
的 size 不斷減一,作為隨機變數的範圍entry
list_del
將尾巴成為獨立的 entry
,將尾巴用 list_add
加到目標 entry
的下一個 entry
,再將目標 entry
移動到尾巴,作為交換Risheng1128
Risheng1128