Serveur multimedia ArtNet
Projet en cours de développement.
Sommaire
Matériel
- 4 x Raspberry Pi 3 :
- 4 x Disque dur WD PiDrive : https://www.wdc.com/fr-fr/products/wdlabs/wd-pidrive-foundation-edition.html#WD2500LMCW
- 4 x Alimentations 3A : https://www.wdc.com/fr-fr/products/wdlabs/wd-pidrive-power-kit-3a.html#WDLB029RNE
- 4 x Boitiers : https://www.wdc.com/fr-fr/products/wdlabs/wd-pidrive-enclosure-square.html#WDLB024RNN
- 1 x Borne d'accès WIFI Netgear WAC104
- 2 x Câbles ethernet Cat 5
- 1 x Node OnPC
- 2 x Adpatateurs USB-C / Ethernet
- 1 x Ordinateur tournant sous Windows
Switch WIFI
https://www.pcastuces.com/pratique/materiel/pont_wifi/page1.htm
https://www.amazon.fr/TP-Link-TL-WA901ND-passive-antenne-démontable/dp/B002YETVXC/?tag=hardwfr-21
http://www.netgear.fr/business/products/wireless/essentials-wireless/WAC104.aspx
https://support.actlighting.com/knowledgeBase/7374771
https://support.actlighting.com/knowledgeBase/13194057
Mettre à jour le firmware
Connecter l'ordinateur au point d'accès
Sur MAC Dans Préférences système > Réseaux, choisissez le port ethernet puis dérouler la liste Configurer IPV4 > Manuellement Ensuite dans remplir les champs suivants :
Adresse IP : 192.168.0.210 Sous-réseau : 255.255.255.0
Cliquer sur Appliquer Allumer la borne d'accès et attendre que le voyant wifi s'allume en vert de manière permanente. Brancher avec un câble RJ45 l'ordinateur et la borne d'accès.
Dans un navigateur entrer l'adresse suivante
192.168.0.100
Entrer les valeurs suivantes dans les champs correspondants :
Admin : admin Password : password
La fenêtre de configuration s'ouvre.
Dans Setup > Wireless Setup
Changer le SSID et le mot de passe pour le 2,4 gHz. Changer le canal de diffusion choisir le 1 car le 12 et le 13 peuvent poser des problèmes.
Dans les options de sécurité choisir :
WPA2-PSK [AES]
Cliquer sur Apply pour sauvegarder la configuration.
Dans Advaced Setup > IP Settings, cocher la case :
Use fixed IP Address (not recommended):
Vous pouvez changer l'adresse IP
IP address : 192.168.1.200 Netmask : 255.255.255.0 Gateway IP address : 192.168.1.1 Primary DNS : 192.168.1.1
Cliquer sur Apply
Installer un disque dur
Pour pouvoir utiliser des fichiers vidéos conséquents, nous allons utiliser le système de fichier exFat. Pour cela, il faut installer le gestionnaires sur la PI
sudo apt-get update sudo apt-get install exfat-fuse
Une fois le disque branché sur un port USB de la Raspberry, taper :
sudo fdisk -l
Vous verrez ainsi que le disque de 250 Go a bien été détecté. Vous devez avoir un résultat semblable :
Disk /dev/sda: 232.9 GiB, 250025607168 bytes, 488331264 sectors
Préparer la partition du disque
sudo cfdisk /dev/sda
Il faut d'abord supprimer la partition existante, si besoin utiliser les flèches Haut et Bas pour vous déplacer et choisir la bonne partition à effacer. Une fois celle-ci sur-lignée, allez sur [ Delete ] avec les flèches Droite et Gauche, puis valider avec Enter.
Créer une nouvelle partition
Allez sur [ New ] et validez. Comme nous n'en créons qu'une seule elle prendra la totalité de l'espace disque, donc valider la taille proposée. Et validez la création d'une partition Primaire. Votre partition st créée et elle s'affiche. Nous allons modifier le type de la partition en allant sur [ Type ] et validez. dans le menu déroulant choisissez le type
7 HPFS/NTFS/exFAT
Pour finaliser, aller sur [ Write ] et validez et confirmez avec yes Le message suivant s'affiche :
The partition table has been altered.
L'opération s'est donc bien déroulée. Pour quitter l'application allez sur [ Quit ] et validez.
Vous pouvez vérifier le résultat avec :
sudo fdisk -l
Formater la partition
Une fois la pétition crée, il faut la formater avec :
sudo mkfs.exfat -n RaspiHDD /dev/sda1
Monter le disque dur
Créer le répertoire dans lequel le disque sera monté par exemple pour qu'il soit facilement accessible, le nom du répertoire sera par exemple PIHDD :
sudo mkdir /home/pi/PIHDD
Monter le disque au point voulu :
sudo mount /dev/sda1 /home/pi/PIHDD
Vérifier que l'opération s'est bien déroulée. Aucune erreur ne devrait apparaître :
ls /home/pi/PIHDD
Monter les disque automatiquement au démarrage
Repérer en premier lieu l'identifiant de la partition (UUID) :
sudo blkid
Par exemple 5C24-1453
Il faut ensuite modifier le fichier fstab
sudo nano /etc/fstab
Ajouter à la fin du fichier la ligne suivante :
UUID=5C24-1453 /home/pi/PIHDD exfat defaults,auto,umask=000,users,rw 0 0
Ctrl + O validez et Ctrl + X pour finaliser l'opération.
Sources
- https://www.raspberrypi.org/documentation/configuration/external-storage.md
- http://www.framboise314.fr/donnez-de-lespace-a-votre-framboise314-un-disque-dur-pour-le-raspberry-pi/
- https://doc.ubuntu-fr.org/exfat
Installer OLA
https://www.openlighting.org/ola/linuxinstall/
Installer les dépendances :
sudo apt-get install autoconf uuid-dev pkg-config libncurses5-dev libtool libcppunit-dev libmicrohttpd-dev automake g++ protobuf-compiler libprotobuf-lite10 python-protobuf libprotobuf-dev libprotoc-dev zlib1g-dev bison flex make libftdi-dev libftdi1 libusb-1.0-0-dev liblo-dev libavahi-client-dev python-numpy
Avec Git
sudo apt-get install git git clone https://github.com/OpenLightingProject/ola.git ola cd ola
Pour installer les fichiers manquants :
autoreconf -i
Pour configurer l'installation
./configure --help
Vous donnera la liste des éléments possible à inclure. Ici nous allons faire simple, nous n'utiliserons que la partie ArtNet
./configure --disable-all-plugins --enable-artnet --enable-http
Compiler en utilisant les quatre coeurs du processeur (durée 25 minutes):
make -j4 make check sudo make install
Pour pouvoir utiliser les nouvelles libraires installées :
sudo ldconfig
Pour lancer le serveur lad
olad -l 3
Vous pouvez à présent vérifier que vous pouvez accéder au serveur en tapant dans votre navigateur préféré :
raspberrypi.local:9090
Une page s'affiche. Il va falloir activer un univers. Cliquer sur le bouton [Add Universe]. Pour notre affaire, il faut remplir les champs
Universe Id : 0 Universe Name : Raspberry0
Et cocher la case
ArtNet [192.168.1.x] Input
Puis valider la création en cliquant sur le bouton [Add Universe]. Désormais il apparait dans les univers actifs.
Pour vérifier qu'il reçoit bien des informations, dans le menu de gauche cliquer sur Raspberry0, puis sur l'onglet [DMXMonitor] Dans cette page vous pouvez voir toutes les valeurs reçues par le serveur OLA.
Configurer OLA
https://www.openlighting.org/ola/advanced-topics/patch-persistency/
Installer OMXPlayer
Si git n'est pas installé :
sudo apt-get install git
Ensuite :
git clone https://github.com/popcornmix/omxplayer.git
Une fois le téléchargement fini
cd omxplayer
Installer les dépendances :
sudo apt-get update && sudo apt-get install git-core libasound2-dev libva1 libpcre3-dev libboost-dev libssl1.0-dev libssh-dev libsmbclient-dev
Le script prépare les fichiers nécessaire à l'installation. S'i manque des dépendances, ce script demande de les installer.
./prepare-native-raspbian.sh
Compiler avec
make ffmpeg make -j4
Puis installer avec :
sudo make install
https://github.com/turingmachine/omxplayer-sync
Programmation C++
https://docs.openlighting.org/ola/doc/0.10.4/namespaceola_1_1client.html
g++ -o multi multi.cpp $(pkg-config --cflags --libs libola)
///////////////////////////////////////////////////////////////////////////////////////////////////
//
// g++ -o multi multi.cpp $(pkg-config --cflags --libs libola)
//
///////////////////////////////////////////////////////////////////////////////////////////////////
#include <iostream>
#include <vector>
#include <string>
#include <algorithm>
#include <dirent.h>
#include <fstream>
#include <ola/DmxBuffer.h>
#include <ola/Logging.h>
#include <ola/client/ClientWrapper.h>
using namespace std;
// Variabes globales
static const string chemin = "/home/pi/PIHDD/";
vector<string> listeFichiers;
int fichirEnLecture = -1;
void RegisterComplete(const ola::client::Result& result) {
if (!result.Success()) {
OLA_WARN << "Failed to register universe: " << result.Error();
}
}
void arreterVideo(int index) {
cout << "-------------------------------------------------" << endl;
cout << "Arrêt vidéo" << endl << endl;
system("killall omxplayer.bin");
}
void demarrerVideo(int index) {
cout << "-------------------------------------------------" << endl;
cout << "Video : " << listeFichiers[index] << endl << endl;
int pid = fork();
if(pid == 0) {
string fichierVideo = chemin + listeFichiers[index];
execlp("omxplayer", " ", fichierVideo.c_str(), NULL);
_exit(0);
}
}
// Appeler lors de la réception d'une trame DMX
void NewDmx(const ola::client::DMXMetadata &metadata,
const ola::DmxBuffer &data) {
if ((fichirEnLecture >= 0) && (data.Get(fichirEnLecture) < 128)) {
arreterVideo(fichirEnLecture);
fichirEnLecture = -1;
}
for (int i = 0; i < listeFichiers.size(); i++) {
if (data.Get(i) > 127) {
if (fichirEnLecture != i) {
arreterVideo(fichirEnLecture);
fichirEnLecture = i;
demarrerVideo(fichirEnLecture);
break;
}
}
}
}
void listerRepertoire(string chemin, vector<string> &tableau) {
DIR* repertoire = opendir(chemin.c_str());
struct dirent* fichierLu;
if (repertoire != NULL) {
while ((fichierLu = readdir(repertoire)) != NULL) {
if (fichierLu->d_name[0] != '.') {
tableau.push_back(fichierLu->d_name);
}
}
sort (tableau.begin(), tableau.end());
}
closedir(repertoire);
}
unsigned int univers() {
unsigned int univers;
ifstream fichier("/home/pi/.ola/ola-port.conf", ios::in);
string ligne;
size_t position;
while(getline(fichier, ligne)) {
position = ligne.find("2-1-I-0 = ");
if (position == 0) {
ligne = ligne.substr(10);
univers = stoi(ligne);
cout << univers << endl;
}
}
fichier.close();
return univers;
}
int main() {
// Lister les fichiers vidéos présents
listerRepertoire("/home/pi/PIHDD", listeFichiers);
for (int i = 0; i < listeFichiers.size(); i++) {
cout << listeFichiers[i] << endl;
}
ola::InitLogging(ola::OLA_LOG_INFO, ola::OLA_LOG_STDERR);
ola::client::OlaClientWrapper wrapper;
if (!wrapper.Setup())
exit(1);
ola::client::OlaClient *client = wrapper.GetClient();
// Set the callback and register our interest in this universe
client->SetDMXCallback(ola::NewCallback(&NewDmx));
client->RegisterUniverse(
univers(), ola::client::REGISTER,
ola::NewSingleCallback(&RegisterComplete));
wrapper.GetSelectServer()->Run();
return 0;
}
Configuration supplémentaires
https://www.supinfo.com/articles/single/2442-raspbian-jessy-configuration-wifi-ligne-commande
https://wiki.openlighting.org/index.php?title=Raspberry_Pi_Media_Player
Écran noir au démarrage
- modifier le fichier cmdline.txt
sudo nano /boot/cmdline.txt
et ajouter sur la même ligne à la fin
consoleblank=20
- source
https://www.raspberrypi.org/documentation/configuration/screensaver.md
Accéder à la borne
Il faut éditer le fichier wpa_supplicant.conf avec la commande :
sudo nano /etc/wpa_supplicant/wpa_supplicant.conf
Ensuite ajouter :
ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=netdev network={ ssid="SSID de la borne" psk="Mot de passe de la borne" key_mgmt=WPA-PSK }
Adresse IP fixe
Afin d'obtenir une IP fixe pour chaque Raspberry Pi, il faut éditer le fichier /etc/network/interfaces avec la commande :
sudo nano /etc/network/interfaces
Puis insérer le lignes suivantes à la fin du fichier :
auto wlan0 iface lo inet loopback iface eth0 inet dhcp allow-hotplug wlan0 iface wlan0 inet static address 192.168.1.101 netmask 255.255.255.0 gateway 192.168.1.200 wpa-conf /etc/wpa_supplicant/wpa_supplicant.conf iface default inet dhcp
GrandMA2
Configuration ArtNet
Configuration sACN
Mettre le node dans le même sous-réseau :
[Setup] > [Network Configuration] > [DMX Node]
Pour modifier l'IP du code faire un clic secondaire sur le champ IP Ethernet 1 (Eth0) et entrer
192.168.1.202
Configurer la sortie sACN
[Setup] > [Network Protocols] > [sACN]
Remplir les champs avec les valeurs indiquées ci-dessous :
Ne pas oublier d'activer la sortie sACN en cliquant sur
[Ouput sACN Active]
Version sACN
Avec l'utilisation des sACN, il n'est pas nécessaire d'installer le serveur OLA. Cela simplifie grandement l'installation et l'exploitation.
Les fichiers sont triés et lus dans l'ordre alphabétique qui correspondent au numéro de circuit.
Le circuite 512 éteint la Raspberry PI. Il est nécessaire de le faire pour ne pas détériorer la Pi.
///////////////////////////////////////////////////////////////////////////////////////////////////
//
// g++ -Wall -o UDP UDP.cpp
// Documentation sACN : http://tsp.esta.org/tsp/documents/docs/E1-31-2016.pdf
// http://bousk.developpez.com/cours/reseau-c++/UDP/01-introduction-premiers-pas/
//
///////////////////////////////////////////////////////////////////////////////////////////////////
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <fcntl.h>
#include <iostream>
#include <string>
#include <sstream> // stringstream
#include <vector>
#include <algorithm>
#include <dirent.h>
#include <fstream>
using namespace std;
// Variabes globales
static const string chemin = "/home/pi/PIHDD/";
static const int port = 5568; // Port par défaut du protocole sACN
vector<string> listeFichiers;
unsigned int fichirEnLecture = -1;
string intToHexString (int n) {
ostringstream oss;
oss << hex << n;
return oss.str();
}
string intToString (int n) {
ostringstream oss;
oss << n;
return oss.str();
}
void arreterVideo(int index) {
cout << "-------------------------------------------------" << endl;
cout << "Arrêt vidéo" << endl << endl;
system("killall omxplayer.bin");
}
void demarrerVideo(int index) {
cout << "-------------------------------------------------" << endl;
cout << "Video : " << listeFichiers[index] << endl << endl;
int pid = fork();
if(pid == 0) {
string fichierVideo = chemin + listeFichiers[index];
execlp("omxplayer", " ", fichierVideo.c_str(), NULL);
_exit(0);
}
}
void arreterRaspberry() {
system("sudo halt");
exit(1);
}
void listerRepertoire(string chemin, vector<string> &tableau) {
DIR* repertoire = opendir(chemin.c_str());
struct dirent* fichierLu;
if (repertoire != NULL) {
while ((fichierLu = readdir(repertoire)) != NULL) {
if (fichierLu->d_name[0] != '.') {
tableau.push_back(fichierLu->d_name);
}
}
sort (tableau.begin(), tableau.end());
}
closedir(repertoire);
}
int main(int argc, char *argv[]) {
// Variables ;
int yes = 1;
// Création du socket
int sckt = socket( AF_INET, SOCK_DGRAM, IPPROTO_UDP );
if (sckt < 0) {
perror("socket");
exit(1);
}
// Ouverture du socket
sockaddr_in adresse;
adresse.sin_family = AF_INET; // L'adresse est IPv4
adresse.sin_port = htons(port); // Définit le port
adresse.sin_addr.s_addr = INADDR_ANY; // Permet d'écouter sur toutes les interfaces locales
// V2rifie si le socket n'est pas déjà ouvert
int res = setsockopt(sckt, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int));
if(res == -1) {
perror("setsockopt");
exit(1);
}
res = bind(sckt, (struct sockaddr*) &adresse, sizeof(adresse));
if (res < 0 ) {
perror("bind");
exit(1);
}
// Lister les fichiers vidéos présents
listerRepertoire(chemin, listeFichiers);
for (unsigned int i = 0; i < listeFichiers.size(); i++) {
cout << listeFichiers[i] << endl;
}
// Boucle principale
while(1) {
char packet_data[638];
struct sockaddr_in cliaddr;
socklen_t taille = sizeof(cliaddr);
int received_bytes = recvfrom(sckt, packet_data, sizeof(packet_data), 0, (struct sockaddr*) &cliaddr, &taille);
if ( received_bytes > 0 ) {
// Traitement des données reçues
for (unsigned int i = 0; i < listeFichiers.size(); i++) {
if (packet_data[i + 126] > 127) { // 126 Début de la trame DMX
if (fichirEnLecture != i) {
arreterVideo(fichirEnLecture);
fichirEnLecture = i;
demarrerVideo(fichirEnLecture);
break;
}
}
}
// Le canal 512 éteint la Raspberry Pi
if (packet_data[637] > 127) {
arreterRaspberry();
}
/*
//Affichage des valeurs
string blip = "";
for (int i = 126; i < 638; i++) { // Les vameurs DMX commencent à partir de l'indice 126 voi p. 5
blip.append(intToString(packet_data[i]) + "/");
}
cout << blip << endl << endl;
*/
}
}
close(sckt);
return 0;
}