2024-01-14 18:37:58 +01:00
---
2024-02-18 15:56:22 +01:00
image_path: "https://community.kde.org/images.community/thumb/a/af/Mascot_konqi-base-plasma.png/250px-Mascot_konqi-base-plasma.png"
2024-02-18 16:05:36 +01:00
slug: kde_first_contrib
2024-02-18 14:03:33 +01:00
title: Mes premières contributions à KDE
2024-02-18 15:56:22 +01:00
date: 2024-01-12T00:00:00Z
2024-02-18 14:03:33 +01:00
description: Ma première configuration et ma première contribution à l'environnement mobile (Plasma-Mobile).
2024-01-14 18:37:58 +01:00
project_link: none
draft: true
tags:
- kde
- linux
---
2024-02-18 14:03:33 +01:00
## Rapide rappel à propos de KDE
2024-01-14 18:37:58 +01:00
KDE est un projet qui a pour but de développer des logiciels open-source. Le projet comporte plusieurs éléments les plus connus.
2024-02-18 14:03:33 +01:00
- [KF5 ou KF6 ](https://develop.kde.org/products/frameworks/ ) est un ensemble de librairies utilisées pour le développement de l'ensemble des logiciels de KDE. Il est basé sur la librairie [QT ](https://www.qt.io/ ).
2024-02-18 15:34:34 +01:00
> _Exemple_ : Kirigami que l'on peut voir comme un "flutter".
2024-02-18 14:03:33 +01:00
>
2024-02-18 15:34:34 +01:00
> _Note:_
2024-02-18 14:03:33 +01:00
>
> KF5 => Basé sur QT5
2024-02-18 15:34:34 +01:00
>
2024-02-18 14:03:33 +01:00
> KF6 => Basé sur QT6
- [Plasma Shell ](https://kde.org/fr/plasma-desktop/ ) est l'environnement de bureau développé pour Linux.
## Ma configuration
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Pour développer sur KDE, KDE fournis un utilitaire assez puissant `kdesrc-build` .
2024-02-18 15:34:34 +01:00
Il permet de configurer l'IDE, compiler, exécuter et gérer les dépendances avec l'aide d'une seule ligne de commande.
2024-01-14 18:37:58 +01:00
2024-02-18 15:34:34 +01:00
Pour le configurer, il suffit de suivre la procédure sur ce [lien ](https://community.kde.org/Get_Involved/development/Set_up_a_development_environment ).
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
### Mes premières difficultés
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Il faut savoir que quand j'ai commencé à vouloir contribuer au projet KDE. Le projet était en train de migrer de QT5 à QT6.
Donc, les configurations nécessaires pour compiler correctement la version 6 était nettement plus compliqué que maintenant.
2024-02-18 15:34:34 +01:00
Il n'y avait aucune documentation pour migrer la configuration fournie de base, il fallait fouiller sur Github ou sur les blogs pour réussir
2024-02-18 14:03:33 +01:00
à avoir la configuration.
2024-02-18 15:34:34 +01:00
Aujourd'hui, il suffit juste de suivre le lien ci-dessus et on peut directement travailler sur KDE 6.
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Pour le développement sur KDE Plasma 6, j'ai dû faire quelques réajustements sur le fichier `~/.config/kdesrc-buildrc` .
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
J'ai changé la branche par défaut avec l'option `branch-group kf6-qt6` .
Elle permet de cloner les repositories correctement et d'ajouter l'option cmake `-DBUILD_WITH_QT6=on` .
J'ai aussi changé les fichiers de configuration utilisées pour obtenir le résultat ci-dessous.
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
```
global
branch-group kf6-qt6
[...]
end global
include ~/kde/src/kdesrc-build/data/build-include/kf6-common-options.ksb
include ~/kde/src/kdesrc-build/data/build-include/kf6-frameworks.ksb
include ~/kde/src/kdesrc-build/data/build-include/kf6-workspace.ksb
include ~/kde/src/kdesrc-build/data/build-include/kf6-applications.ksb
include ~/kde/src/kdesrc-build/data/build-include/kf6-kdepim.ksb
```
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
### Configurations utiles
2024-01-14 18:37:58 +01:00
2024-02-18 15:34:34 +01:00
#### Désactiver l'arrêt de la compilation lors d'une erreur de compilation.
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
En général, `kdesrc-build` s'adapte et retire tous les projets qui dépendent de la dépendance qui n'a pas réussi à compiler.
Le projet était encore en cours de développement donc les erreurs de compilations étaient très régulières.
J'ai tout simplement désactivé avec l'option `stop-on-failure false` .
2024-01-14 18:37:58 +01:00
2024-02-18 15:34:34 +01:00
#### Activer le support des LSP
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
C'est très pratique si on utilise VSCode ou même n'importe quel outil qui supporte les [LSP ](https://microsoft.github.io/language-server-protocol/ ).
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Il suffit d'activer les options:
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
* `compile-commands-linking` => `true`
* `compile-commands-export` => `true`
2024-01-14 18:37:58 +01:00
2024-02-18 15:34:34 +01:00
#### Générer les projets VSCode
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Il va générer automatiquement les workspaces VSCode avec les extensions recommander et le paramètrage par défaut.
Ça permet de travailler dans de très bonne condition avec VSCode.
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Il suffit de l'activer avec `generate-vscode-project-config true`
2024-01-14 18:37:58 +01:00
2024-02-18 15:34:34 +01:00
#### Options utiles pour la commande kdesrc-build
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
* `-D` : Pour ne pas inclure les dépendances dans la compilation
> Exemple: kdesrc-build -D kate
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
* `-S` : Pour ne pas mettre à jour les sources
> Exemple: kdesrc-build -S kate
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
## Ma contribution
2024-01-14 18:37:58 +01:00
2024-02-18 15:34:34 +01:00
J'ai contribué sur le support de la lampe torche pour l'environnement Plasma Mobile.
La lampe torche fonctionnait uniquement sur les [pinephones ](https://pine64.org/devices/pinephone/ ) en utilisant le fichier en dure `/sys/devices/platform/led-controller/leds/white:flash/brightness`
2024-02-18 14:03:33 +01:00
([Source ](https://invent.kde.org/plasma/plasma-mobile/-/blob/f970aa7acf2e9794ab4ed6b75e8f549bece83561/quicksettings/flashlight/flashlightutil.cpp#L17 )).
2024-02-18 15:34:34 +01:00
J'ai réécris le module pour que ça fonctionne de manière générale sur tous les téléphones.
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
### Rappel sur le fonctionnement du noyau Linux autour des leds
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
Pour comprendre comment la fonctionnalité, un petit rappel s'impose autour du fonctionnement des leds avec le noyau Linux.
2024-02-18 15:34:34 +01:00
Dès qu'une led est détecté par le noyau, il va mettre à disposition plusieurs fichiers.
On les trouvent dans le dossier correspondant au driver et aussi à l'emplacement physique sur la carte mère ou sur le SOC.
Mais on peut aussi les trouver plus facilement dans le répertoire `/sys/class/leds/` .
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
> sys => Fichiers système
> class => Trier par class.
> leds => La class `leds`. On peut trouver aussi par appareil `bluetooth`, puce `tpm`, ...
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
On disposent de plusieurs fichiers importants :
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
* `brightness` : Permet de changer ou de récupérer l'intensité de la led actuelle.
* `max_brightness` : Permet de connaitre l'intensité max de la led.
* `color` : Ce fichier n'existe pas tout le temps mais permet d'avoir la couleur de la led.
* Le reste des fichiers ne sont pas important pour nous.
### Explication du fonctionnement de mon code petit à petit
2024-02-18 15:34:34 +01:00
Tout d'abord, il faut savoir que j'utilise la libraire Udev. Il est prévu que je migre vers la librairie [Solid ](https://invent.kde.org/frameworks/solid ) une fois le support des leds ajoutées.
2024-02-18 14:03:33 +01:00
Udev pour faire simple, c'est une librairie qui permet d'interagir avec les appareils assez directement sans intermédiaire.
2024-02-18 15:34:34 +01:00
La librairie permet également de lister, filtrer, détecter si un appareil est connecté/déconnecté, ....
2024-02-18 14:03:33 +01:00
Pour commencer, je dois d'abord instancier Udev avec pour l'utiliser par la suite.
```cpp
struct udev *udev = udev_new();
```
2024-02-18 15:34:34 +01:00
Ensuite, je liste tous les périphériques qui m'intéressent donc ceux appartenant à la class `leds` .
Pour ça, il me suffit de créer un énumerateur sur lequel on appliquera tous les filtres.
> On peut voir une class comme une catégorie d'appareils.
2024-02-18 14:03:33 +01:00
```cpp
struct udev_enumerate *enumerate = udev_enumerate_new(udev);
```
2024-02-18 15:34:34 +01:00
J'applique le filtre pour lister uniquement ceux qui appartiennent à la class `leds` .
2024-02-18 14:03:33 +01:00
```cpp
#define TORCH_SUBSYSTEM "leds"
...
udev_enumerate_add_match_subsystem(enumerate, TORCH_SUBSYSTEM);
```
2024-02-18 15:34:34 +01:00
J'applique le filtre pour lister uniquement ceux qui ont dans le nom `flash` ou `torch` .
2024-02-18 14:03:33 +01:00
```cpp
udev_enumerate_add_match_sysname(enumerate, "*:torch");
udev_enumerate_add_match_sysname(enumerate, "*:flash");
```
Puis je lance le scan avec mes filtres.
```cpp
udev_enumerate_scan_devices(enumerate);
```
Au début, il faut savoir que je prenais le premier appareil avec le code ci-dessous mais certain téléphone ont une torche de couleur jaune et blanche.
2024-02-18 15:34:34 +01:00
Certain utilisateur malheureusement ne pouvait pas utiliser la fonctionnalité.
2024-02-18 14:03:33 +01:00
```cpp
struct udev_list_entry *devices = udev_enumerate_get_list_entry(enumerate);
struct udev_list_entry *entry = udev_list_entry_get_next(devices);
```
[Source ](https://invent.kde.org/plasma/plasma-mobile/-/blob/5c6a97caa52d549c0cb02b17fc65a3a7d729d237/quicksettings/flashlight/flashlightutil.cpp )
2024-02-18 15:34:34 +01:00
Donc, j'ai changé le code avec le code ci-dessous qui permet d'itérer sur tous les appareils.
2024-02-18 14:03:33 +01:00
```cpp
struct udev_list_entry *devices = udev_enumerate_get_list_entry(enumerate);
struct udev_list_entry *entry = nullptr;
struct udev_device *device = nullptr;
udev_list_entry_foreach(entry, devices)
{
[...]
}
```
Pour chaque appareil, j'ai récuperé le répertoire ou se situe l'appareil avec le code ci-dessous
qui me renvoie par exemple: `/sys/devices/platform/led-controller/leds/white:flash` .
```cpp
const char *path = udev_list_entry_get_name(entry);
```
J'instancie une instance udev_device pour interagir avec lui depuis le répertoire.
```cpp
struct udev_device *device = udev_device_new_from_syspath(udev, path);
```
Et à partir de là, il me suffit de lire les fichiers dont j'ai besoin.
```cpp
const char *maxBrightness = udev_device_get_sysattr_value(device, "[NOM DU FICHIER]");
2024-01-14 18:37:58 +01:00
```
2024-02-18 14:03:33 +01:00
Je lis d'abord le fichier 'color' et je regarde si la couleur est blanche sinon je continue.
Si je ne trouve pas de led de couleur blanche, je prends la dernière valeur récupérée.
2024-02-18 15:34:34 +01:00
Une fois pour chaque appareil trouvé, il me reste plus qu'à récupérer les infos dont j'ai besoin.
Donc, la luminosité maximale `max_brightness` et la luminosité actuelle `brightness` pour correctement initialiser l'interface de Plasma Mobile.
2024-02-18 14:03:33 +01:00
Quand je souhaite changer la valeur, je dois d'abord convertir la valeur entière en chaîne de charactère.
```cpp
const_cast< char * > (m_torchEnabled ? "0" : m_maxBrightness)
```
Ensuite, je change la luminosité
```cpp
udev_device_set_sysattr_value(m_device, "brightness", nouvelle_valeur_en_char_*);
```
### Fichier final
2024-02-18 15:34:34 +01:00
[Source ](https://invent.kde.org/plasma/plasma-mobile/-/blob/d162f96a63600d5b45bb8294afdb84efd85833f9/quicksettings/flashlight/flashlightutil.cpp )
2024-02-18 14:03:33 +01:00
```cpp
/*
* SPDX-FileCopyrightText: 2020 Han Young < hanyoung @protonmail .com >
* SPDX-FileCopyrightText: 2022 by Devin Lin < devin @kde .org >
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "flashlightutil.h"
#include <cstring>
#include <fcntl.h>
#include <libudev.h>
#include <unistd.h>
#include <QDebug>
#include <QFileInfo>
#define TORCH_SUBSYSTEM "leds"
FlashlightUtil::FlashlightUtil(QObject *parent)
: QObject{parent}
, m_device{nullptr}
, m_isAvailable{false}
{
findTorchDevice();
}
FlashlightUtil::~FlashlightUtil()
{
if (m_device != nullptr) {
udev_device_unref(m_device);
}
}
void FlashlightUtil::toggleTorch()
{
if (!isAvailable()) {
qWarning() < < "Flashlight not available";
return;
}
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
int ret = udev_device_set_sysattr_value(m_device, "brightness", const_cast< char * > (m_torchEnabled ? "0" : m_maxBrightness));
if (ret < 0 ) {
qWarning() < < "Flashlight can't be toggled";
return;
}
2024-01-14 18:37:58 +01:00
2024-02-18 14:03:33 +01:00
m_torchEnabled = !m_torchEnabled;
Q_EMIT torchChanged(m_torchEnabled);
}
bool FlashlightUtil::torchEnabled() const
{
return m_torchEnabled;
}
bool FlashlightUtil::isAvailable() const
{
return m_isAvailable;
}
void FlashlightUtil::findTorchDevice()
{
if (m_device != nullptr) {
udev_device_unref(m_device);
}
m_device = nullptr;
m_isAvailable = false;
struct udev *udev = udev_new();
struct udev_enumerate *enumerate = udev_enumerate_new(udev);
udev_enumerate_add_match_subsystem(enumerate, TORCH_SUBSYSTEM);
udev_enumerate_add_match_sysname(enumerate, "*:torch");
udev_enumerate_add_match_sysname(enumerate, "*:flash");
udev_enumerate_scan_devices(enumerate);
struct udev_list_entry *devices = udev_enumerate_get_list_entry(enumerate);
struct udev_list_entry *entry = nullptr;
struct udev_device *device = nullptr;
udev_list_entry_foreach(entry, devices)
{
const char *path = udev_list_entry_get_name(entry);
if (path == nullptr) {
continue;
}
if (device != nullptr) {
udev_device_unref(device); // Use to free memory from previous loop iteration
}
device = udev_device_new_from_syspath(udev, path);
if (device == nullptr) {
continue;
}
qInfo() < < "Found flashlight device : " < < path ;
const char *color = udev_device_get_sysattr_value(device, "color");
if (color == nullptr) {
continue;
}
qInfo() < < "Flash color : " < < color ;
if (std::strcmp(color, "white") == 0) {
break;
}
}
if (device == nullptr) {
qWarning() < < "No flashlight device found";
return;
}
const char *maxBrightness = udev_device_get_sysattr_value(device, "max_brightness");
if (maxBrightness == nullptr) {
qWarning() < < "Failed to read max_brightness from udev device";
return;
}
qInfo() < < "Flash maxBrightness : " < < maxBrightness ;
const char *brightness = udev_device_get_sysattr_value(device, "brightness");
if (brightness == nullptr) {
qWarning() < < "Failed to read brightness from udev device";
return;
}
qInfo() < < "Flash brightness : " < < brightness ;
m_maxBrightness = maxBrightness;
m_device = device;
m_isAvailable = true;
m_torchEnabled = std::strcmp(brightness, "0") != 0;
udev_enumerate_unref(enumerate);
udev_unref(udev);
}
```