Příspěvek na blogu Ubuntu upozorňuje na několik zranitelností v rozšíření Linuxu o mandatorní řízení přístupu AppArmor. Společně jsou označovány jako CrackArmor. Objevila je společnost Qualys (technické detaily). Neprivilegovaný lokální uživatel se může stát rootem. Chyba existuje od roku 2017. Doporučuje se okamžitá aktualizace. Problém se týká Ubuntu, Debianu nebo SUSE. Red Hat nebo Fedora pro mandatorní řízení přístupu používají SELinux.
Byla vydána nová verze 19 integrovaného vývojového prostředí (IDE) Qt Creator. Podrobný přehled novinek v changelogu.
Bitwig Studio (Wikipedie) bylo vydáno ve verzi 6. Jedná se o proprietární multiplatformní (macOS, Windows, Linux) digitální pracovní stanici pro práci s audiem (DAW).
Společnost Igalia představila novou linuxovou distribuci (framework) s názvem Moonforge. Jedná se o distribuci určenou pro vestavěné systémy. Vychází z projektů Yocto a OpenEmbedded.
Google Chrome 146 byl prohlášen za stabilní. Nejnovější stabilní verze 146.0.7680.71 přináší řadu novinek z hlediska uživatelů i vývojářů. Podrobný přehled v poznámkách k vydání. Opraveno bylo 29 bezpečnostních chyb. Vylepšeny byly také nástroje pro vývojáře.
D7VK byl vydán ve verzi 1.5. Jedná se o fork DXVK implementující překlad volání Direct3D 3 (novinka), 5, 6 a 7 na Vulkan. DXVK zvládá Direct3D 8, 9, 10 a 11.
Bylo vydáno Eclipse IDE 2026-03 aneb Eclipse 4.39. Představení novinek tohoto integrovaného vývojového prostředí také na YouTube.
Ze systému Slavia pojišťovny uniklo přibližně 150 gigabajtů citlivých dat. Jedná se například o pojistné dokumenty, lékařské záznamy nebo přímou komunikaci s klienty. Za únik může chyba dodavatelské společnosti.
Sněmovna propustila do dalšího kola projednávání vládní návrh zákona o digitální ekonomice, který má přinést bezpečnější on-line prostředí. Reaguje na evropské nařízení DSA o digitálních službách a upravuje třeba pravidla pro on-line tržiště nebo sociální sítě a má i víc chránit děti.
Meta převezme sociální síť pro umělou inteligenci (AI) Moltbook. Tvůrci Moltbooku – Matt Schlicht a Ben Parr – se díky dohodě stanou součástí Meta Superintelligence Labs (MSL). Meta MSL založila s cílem sjednotit své aktivity na poli AI a vyvinout takovou umělou inteligenci, která překoná lidské schopnosti v mnoha oblastech. Fungovat by měla ne jako centralizovaný nástroj, ale jako osobní asistent pro každého uživatele.
Nezatracoval bych popen(), myslím, že to také není špatné řešení. Tak proč mu to rovnou nepředáš jako argument, když se jedná jen o pár dat. Kdyžtak víc rozepiš o co konkrétně jde, třeba nás toho pak napadne víc.
I při použití fork() a zabití rodiče dle mého názoru pokračuje potomek v pohodě, stačí postupovat dle Linux Daemon Writing HOWTO
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <errno.h>
#include <unistd.h>
#include <syslog.h>
#include <string.h>
int main(void) {
/* Our process ID and Session ID */
pid_t pid, sid;
/* Fork off the parent process */
pid = fork();
if (pid < 0) {
exit(EXIT_FAILURE);
}
/* If we got a good PID, then
we can exit the parent process. */
if (pid > 0) {
exit(EXIT_SUCCESS);
}
/* Change the file mode mask */
umask(0);
/* Open any logs here */
/* Create a new SID for the child process */
sid = setsid();
if (sid < 0) {
/* Log the failure */
exit(EXIT_FAILURE);
}
/* Change the current working directory */
if ((chdir("/")) < 0) {
/* Log the failure */
exit(EXIT_FAILURE);
}
/* Close out the standard file descriptors */
close(STDIN_FILENO);
close(STDOUT_FILENO);
close(STDERR_FILENO);
/* Daemon-specific initialization goes here */
/* The Big Loop */
while (1) {
/* Do some task here ... */
sleep(30); /* wait 30 seconds */
}
exit(EXIT_SUCCESS);
}
Jinak proč před fork() nepřipravíš data do paměti, jejich kopie bude dostupná i v potomku, včetně stejné adresy.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#define STRING_MAX 512
int main(int argc, char *argv[])
{
char string[STRING_MAX+1];
pid_t pid;
strcpy(string, "Hello World");
pid = fork();
if (pid < 0) {
exit(EXIT_FAILURE);
}
else if (pid > 0) {
memset(string, 0, STRING_MAX+1);
printf("0x%x parent string = '%s'\n", string, string);
printf("Exit the parent process\n");
exit(EXIT_SUCCESS);
}
sleep(3);
printf("0x%x child string = '%s'\n", string, string);
return 0;
}
$ ./example01
0xbfa67f1f parent string = ''
Exit the parent process
0xbfa67f1f child string = 'Hello World'
Nebo mu to šoupni takhle, když se Ti nechce vytvářet rouru:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(int argc, char *argv[])
{
system("echo 'ABC' |sed 's/ABC/abc/'");
return 0;
}
Něco jako:
if (fork() == 0)
{
char data[25]="Heslo";
fprintf (stdin,"%s",data);
exec ('/usr/bin/cat');
}
Případně méně šílená varianta:
if (fork() == 0)
{
char data[25]="Heslo";
FILE *f;
f = fmemopen(data, 25, "r");
freopen (f,"r",stdin);
exec ('/usr/bin/cat');
}
Tady je problém s tím, že freopen() potřebuje cestu k souboru, takže mu nemůžu podstrčit memstream...
pro duplikaci deskriptoru jsou funkce dup/dup2.
dup2(fd, 0); // fd -> 0
exec(...);
Tohle by se Ti mohlo líbit. Stejný příklad je v knížce "Linux začínáme programovat" v kapitole "Vzájemná komunikace mezi procesy: Roury" a části "Roury a funkce dup". Je to tam i pěkně vysvětleno.
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#define STRING_MAX 512
int main(int argc, char *argv[])
{
char string[STRING_MAX+1];
int pipes[2];
pid_t pid;
strcpy(string, "Hello World");
if (pipe(pipes) == 0) {
pid = fork();
if (pid < 0) {
exit(EXIT_FAILURE);
}
else if (pid > 0) {
close(pipes[0]);
write(pipes[1], string, strlen(string));
close(pipes[1]);
}
close(0);
dup(pipes[0]);
close(pipes[0]);
close(pipes[1]);
execlp("cat", "cat");
}
return 0;
}
Zásadní jsou tyto dva řádky:
close(0);
dup(pipes[0]);
Nejprve se zavře stdin a díky tomu pak následně volaná fce vytvoří kopii deskriptoru nastavenou na 0, protože je nový deskriptor vždy nejnižší možné číslo. V případě fce dup2() je nastaven na hodnotu předanou v druhém argumentu nebo nejbližší větší, v případě, že požadovaná hodnota je již využívána.
Diky za pomoc...
Díky moc za vyvedení z omylu. Máš pravdu, zmátla mě formulace v knize, jak v originále, tak překladu.
... zatímco v případě funkce dup2 je nový deskriptor shodný nebo první volný, který je větší než parametr newfd ...
... while in the case of dup2 it's the same as, or the first available descriptor greater than, the parameter newfd ...
Takže jim evidentně zmizel kus textu a mělo tam být něco jako:
If newfd is negative or greater than or equal to OPEN_MAX, the dup2() function returns a value of -1 and sets errno to EBADF.
V manuálové stránce to řekli hezky na rovinu:
dup2() makes newfd be the copy of oldfd, closing newfd first if necessary.
The call:
fid = dup2(fildes, fildes2);
is equivalent to:
close(fildes2);
fid = fcntl(fildes, F_DUPFD, fildes2);
with the following exceptions:
If fildes2 is negative or greater than or equal to OPEN_MAX, the dup2() function returns a value of -1 and sets errno to EBADF.
If fildes is a valid file descriptor and is equal to fildes2, the dup2() function returns fildes2 without closing it.
If fildes is not a valid file descriptor, dup2() fails and does not close fildes2.
If successful, dup2() returns a value that is equal to the value of fildes2.
If a failure occurs, it returns a value of -1.
Tiskni
Sdílej: