aboutsummaryrefslogtreecommitdiff
path: root/ft_list_remove_if.s
blob: 7ce06a04f216a40da56fce16e8fd45dacae326a6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
; **************************************************************************** ;
;                                                                              ;
;                                                         :::      ::::::::    ;
;    ft_list_remove_if.s                                :+:      :+:    :+:    ;
;                                                     +:+ +:+         +:+      ;
;    By: cacharle <marvin@42.fr>                    +#+  +:+       +#+         ;
;                                                 +#+#+#+#+#+   +#+            ;
;    Created: 2019/11/25 04:23:22 by cacharle          #+#    #+#              ;
;    Updated: 2019/11/25 04:23:22 by cacharle         ###   ########.fr        ;
;                                                                              ;
; **************************************************************************** ;

%ifdef __LINUX__
    %define M_FT_LIST_REMOVE_IF ft_list_remove_if
    %define M_FREE free
%else
    %define M_FT_LIST_REMOVE_IF _ft_list_remove_if
    %define M_FREE _free
%endif

global M_FT_LIST_REMOVE_IF

extern M_FREE

%define NULL 0x0

%macro EXTERN_FUNCTION_SAVE 0
	push rdi
	push rsi
	push rdx
	push rcx
%endmacro

%macro EXTERN_FUNCTION_SAVE_END 0
	pop  rcx
	pop  rdx
	pop  rsi
	pop  rdi
%endmacro

; ft_list_remove_if(t_list **begin_list, void *data_ref,
;                   int (*cmp)(void *data, void *data_ref),
;                   void (*free_fct)(void *))
M_FT_LIST_REMOVE_IF:
	; t_list *saved_next

	; === prolog ===
	push rbp
	mov  rbp, rsp
	sub  rsp, 8

	; === base condition ===
	cmp  rdi, NULL
	je   FT_LIST_REMOVE_IF_END
	cmp  qword [rdi], NULL
	je   FT_LIST_REMOVE_IF_END

	; === compare (*begin_list)->data and data_ref
	EXTERN_FUNCTION_SAVE
	mov  rdi, [rdi]
	mov  rdi, [rdi + 0]
	call rdx                         ; cmp((*begin_list)->data, data_ref)
	EXTERN_FUNCTION_SAVE_END
	cmp  rax, 0
	je   FT_LIST_REMOVE_IF_REMOVE

	; === next element ===
	push rdi
	mov  rdi, [rdi]
	lea  rdi, [rdi + 8]
	call M_FT_LIST_REMOVE_IF
	pop  rdi
	jmp  FT_LIST_REMOVE_IF_END

	; === remove head and go next ===
FT_LIST_REMOVE_IF_REMOVE:
	mov  rax, [rdi]                    ; rax = *begin_list
	mov  rax, [rax + 8]                ; rax = rax->next
	mov  [rbp - 8], rax                ; saved_next = (*begin_list)->next

	push rdi                           ; strange behavior
	EXTERN_FUNCTION_SAVE
	mov  rdi, [rdi]
	mov  rdi, [rdi + 0]
	call rcx                           ; free_fct((*begin_list)->data)
	EXTERN_FUNCTION_SAVE_END
	pop  rdi

	EXTERN_FUNCTION_SAVE
	mov  rdi, [rdi]
	call M_FREE wrt ..plt              ; free(*begin_list)
	EXTERN_FUNCTION_SAVE_END

	mov rax, [rbp - 8]
	mov [rdi], rax
	call M_FT_LIST_REMOVE_IF

FT_LIST_REMOVE_IF_END:
	; === epilog ===
	mov  rsp, rbp
	pop  rbp
	ret