aboutsummaryrefslogtreecommitdiff
path: root/philo_one/src/philo.c
blob: 8a41fe775bd2b1e6acbc18514abfc82646cf61c4 (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
/* ************************************************************************** */
/*                                                                            */
/*                                                        :::      ::::::::   */
/*   philo.c                                            :+:      :+:    :+:   */
/*                                                    +:+ +:+         +:+     */
/*   By: cacharle <marvin@42.fr>                    +#+  +:+       +#+        */
/*                                                +#+#+#+#+#+   +#+           */
/*   Created: 2020/02/09 23:47:14 by cacharle          #+#    #+#             */
/*   Updated: 2021/01/10 09:49:48 by cacharle         ###   ########.fr       */
/*                                                                            */
/* ************************************************************************** */

#include "philo_one.h"

t_philo	*philos_new(t_philo_conf *conf, pthread_mutex_t *forks)
{
	long int	i;
	t_philo		*philos;

	if (conf->philo_num < 0)
		return (NULL);
	if ((philos = malloc(conf->philo_num * sizeof(t_philo))) == NULL)
		return (NULL);
	i = -1;
	while (++i < conf->philo_num)
	{
		philos[i].id = i + 1;
		philos[i].conf = conf;
		philos[i].fork_left = forks + i % conf->philo_num;
		philos[i].fork_right = forks + (i + 1) % conf->philo_num;
		pthread_mutex_init(&philos[i].mutex_start, NULL);
		pthread_mutex_lock(&philos[i].mutex_start);
		pthread_mutex_init(&philos[i].mutex_eat, NULL);
	}
	return (philos);
}

bool	philos_start(t_philo *philos, long int num)
{
	long int	i;

	i = -1;
	while (++i < num)
	{
		if (pthread_create(
				&philos[i].thread,
				NULL,
				(t_routine)routine_philo,
				(void*)(philos + i)) != 0)
		{
			while (--i >= 0)
				pthread_detach(philos[i].thread);
			return (false);
		}
	}
	i = -1;
	while (++i < num)
	{
		if (i % 2 == 0)
			continue;
		pthread_mutex_unlock(&philos[i].mutex_start);
	}
	usleep(1000);
	i = -1;
	while (++i < num)
	{
		if (i % 2 == 1)
			continue;
		pthread_mutex_unlock(&philos[i].mutex_start);
	}

	return (true);
}

void	philos_detach(t_philo *philos, long int num)
{
	long int	i;

	if (philos == NULL)
		return ;
	i = -1;
	while (++i < num)
		pthread_detach(philos[i].thread);
}