summaryrefslogtreecommitdiffstats
path: root/src/e_builtins.c
blob: e35416a9791497629b37da68e5476f0782719817 (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
/* ************************************************************************** */
/*                                                                            */
/*                                                        :::      ::::::::   */
/*   e_builtins.c                                       :+:      :+:    :+:   */
/*                                                    +:+ +:+         +:+     */
/*   By: rbousset <marvin@42.fr>                    +#+  +:+       +#+        */
/*                                                +#+#+#+#+#+   +#+           */
/*   Created: 2020/02/14 17:19:27 by rbousset          #+#    #+#             */
/*   Updated: 2020/02/14 17:19:29 by rbousset         ###   ########lyon.fr   */
/*                                                                            */
/* ************************************************************************** */

#include <sys/wait.h>
#include <libft.h>
#include <stdlib.h>
#include <stdint.h>
#include <unistd.h>

#include "m_redirs.h"
#include "s_destroy.h"
#include "s_lcom.h"
#include "s_struct.h"

static void
	e_builtin_child(const t_lcom *ptr,
					uint8_t bu_id,
					t_msh *msh)
{
	int32_t	ret;

	dup_redirs(ptr, msh);
	ret = msh->bu_ptr[bu_id](ptr->argv + 1, msh);
	lcom_clear(&msh->curr);
	s_destroy(msh);
	exit(ret);
}

static void
	e_builtin_parent(pid_t pid,
						const t_lcom *ptr,
						uint8_t bu_id,
						t_msh *msh)
{
	int32_t	status;
	int32_t	ret;

	while (wait(&status) != pid)
		;
	ret = WEXITSTATUS(status);
	if (bu_id != 6)
		msh->ret = ret;
	if (bu_id == 1 && msh->ret == 0)
	{
		msh->bu_ptr[bu_id](ptr->argv + 1, msh);
		/* TODO: export $PWD */
	}
	else if (bu_id == 6 && ret == 0)
	{
		if (ptr->argv[1])
			ret = ft_atoi(ptr->argv[1]);
		else
			ret = msh->ret;
		lcom_clear(&msh->curr);
		s_destroy(msh);
		exit(ret);
	}
}

void
	e_builtin(const t_lcom *ptr,
				uint8_t bu_id,
				t_msh *msh)
{
	pid_t	pid;

	/* TODO: find a way to handle exit | bu_id = 6 */
	if ((pid = fork()) == 0)
	{
		e_builtin_child(ptr, bu_id, msh);
	}
	else if (pid < 0)
	{
		/* TODO: handle fork failed */
	}
	else
	{
		e_builtin_parent(pid, ptr, bu_id, msh);
	}
}