De exec-systeemaanroep wordt gebruikt om een bestand uit te voeren dat zich in een actief proces bevindt. Wanneer exec wordt aangeroepen, wordt het vorige uitvoerbare bestand vervangen en wordt het nieuwe bestand uitgevoerd.
Meer precies, we kunnen zeggen dat het gebruik van exec-systeemaanroep het oude bestand of programma uit het proces zal vervangen door een nieuw bestand of programma. De volledige inhoud van het proces wordt vervangen door een nieuw programma.
Het gebruikersgegevenssegment dat de systeemaanroep exec() uitvoert, wordt vervangen door het gegevensbestand waarvan de naam in het argument wordt gegeven tijdens het aanroepen van exec().
Het nieuwe programma wordt in dezelfde procesruimte geladen. Het huidige proces is gewoon omgezet in een nieuw proces en daarom is de proces-ID PID niet gewijzigd, dit komt omdat we geen nieuw proces maken, we vervangen gewoon een proces door een ander proces in exec.
Als het huidige lopende proces meer dan één thread bevat, worden alle threads beëindigd en wordt de nieuwe procesafbeelding geladen en vervolgens uitgevoerd. Er zijn geen destructorfuncties die threads van het huidige proces beëindigen.
De PID van het proces wordt niet gewijzigd, maar de gegevens, code, stapel, heap, enz. van het proces worden gewijzigd en vervangen door die van het nieuw geladen proces. Het nieuwe proces wordt uitgevoerd vanaf het ingangspunt.
Exec-systeemaanroep is een verzameling functies en in de programmeertaal C zijn de standaardnamen voor deze functies als volgt:
- execl
- voorbeeld
- execlp
- execv
- uitvoerend
- execvp
Hierbij moet worden opgemerkt dat deze functies dezelfde basis hebben uitvoerend gevolgd door een of meer letters. Deze worden hieronder toegelicht:
En: Het is een reeks aanwijzers die naar omgevingsvariabelen verwijst en expliciet wordt doorgegeven aan het nieuw geladen proces.
de: l is voor de opdrachtregelargumenten die een lijst aan de functie hebben doorgegeven
P: p is de padomgevingsvariabele die helpt bij het vinden van het bestand dat is doorgegeven als argument om in het proces te worden geladen.
v: v is voor de opdrachtregelargumenten. Deze worden doorgegeven als een array van verwijzingen naar de functie.
Waarom wordt exec gebruikt?
exec wordt gebruikt wanneer de gebruiker een nieuw bestand of programma in hetzelfde proces wil starten.
Innerlijke werking van exec
Overweeg de volgende punten om de werking van exec te begrijpen:
- De huidige procesafbeelding wordt overschreven door een nieuwe procesafbeelding.
- Nieuwe procesafbeelding is degene die u hebt doorgegeven als exec-argument
- Het huidige lopende proces is beëindigd
- Nieuwe procesafbeelding heeft dezelfde proces-ID, dezelfde omgeving en dezelfde bestandsdescriptor (omdat proces niet wordt vervangen procesafbeelding wordt vervangen)
- De CPU-statistieken en het virtuele geheugen worden beïnvloed. Virtuele geheugentoewijzing van de huidige procesafbeelding wordt vervangen door virtueel geheugen van de nieuwe procesafbeelding.
Syntaxis van exec-familiefuncties:
Hieronder volgen de syntaxis voor elke functie van exec:
int execl(const char* pad, const char* arg, …)int execlp(const char* bestand, const char* arg, …)
int execle(const char* pad, const char* arg, …, char* const envp[])
int execv(const char* pad, const char* argv[])
int execvp(const char* bestand, const char* argv[])
int execvpe(const char* bestand, const char* argv[], char *const envp[])
Beschrijving:
Het retourtype van deze functies is Int. Wanneer de procesafbeelding met succes is vervangen, wordt er niets teruggegeven aan de aanroepende functie omdat het proces dat deze heeft aangeroepen niet langer actief is. Maar als er een fout is, wordt -1 geretourneerd. Als er een fout is opgetreden, foutje is ingesteld.
In de syntaxis:
- pad wordt gebruikt om de volledige padnaam op te geven van het bestand dat moet worden uitgevoerd.
- boos is het argument doorgegeven. Het is eigenlijk de naam van het bestand dat tijdens het proces wordt uitgevoerd. Meestal is de waarde van arg en pad hetzelfde.
- const char* arg in de functies execl(), execlp() en execle() wordt beschouwd als arg0, arg1, arg2, ..., argn. Het is in feite een lijst met verwijzingen naar null-beëindigde tekenreeksen. Hier verwijst het eerste argument naar de bestandsnaam die zal worden uitgevoerd zoals beschreven in punt 2.
- envp is een array die pointers bevat die verwijzen naar de omgevingsvariabelen.
- het dossier wordt gebruikt om de padnaam op te geven die het pad van het nieuwe procesbeeldbestand zal identificeren.
- De functies van exec noemen die eindigen op En worden gebruikt om de omgeving voor de nieuwe procesafbeelding te wijzigen. Deze functies geven een lijst met omgevingsinstellingen door met behulp van het argument envp . Dit argument is een array van tekens die verwijst naar een null-beëindigde String en de omgevingsvariabele definieert.
Om de exec-familiefuncties te gebruiken, moet u het volgende headerbestand in uw C-programma opnemen:
#erbij betrekkenVoorbeeld 1: Exec-systeemaanroep gebruiken in C-programma
Beschouw het volgende voorbeeld waarin we exec-systeemaanroep hebben gebruikt in C-programmering in Linux, Ubuntu: We hebben hier twee c-bestanden example.c en hello.c:
voorbeeld.c
CODE:
#erbij betrekken#erbij betrekken
#erbij betrekken
inthoofd(intargc, char *argv[])
{
printf ('PID van voorbeeld.c = %dN',getpid());
char *argumenten[] = {'Hallo', 'C', 'Programmeren',NUL};
execv('./Hallo',argumenten);
printf ('Terug naar voorbeeld.c');
opbrengst 0;
}
hallo.c
CODE:
#erbij betrekken#erbij betrekken
#erbij betrekken
inthoofd(intargc, char *argv[])
{
printf ('We zijn in Hallo.cN');
printf ('PID van hallo.c = %dN',getpid());
opbrengst 0;
}
UITGANG:
PID van voorbeeld.c = 4733We zijn in Hello.c
PID van hallo.c = 4733
In het bovenstaande voorbeeld hebben we een voorbeeld.c-bestand en hallo.c-bestand. In het voorbeeld .c bestand hebben we allereerst de ID van het huidige proces afgedrukt (bestand voorbeeld.c draait in huidig proces). Vervolgens hebben we in de volgende regel een reeks tekenaanwijzers gemaakt. Het laatste element van deze array moet NULL zijn als het eindpunt.
Vervolgens hebben we de functie execv() gebruikt die de bestandsnaam en de tekenaanwijzerarray als argument heeft. Hier moet worden opgemerkt dat we ./ hebben gebruikt met de naam van het bestand, het specificeert het pad van het bestand. Aangezien het bestand zich in de map bevindt waar example.c zich bevindt, is het dus niet nodig om het volledige pad op te geven.
Wanneer de functie execv() wordt aangeroepen, wordt onze procesafbeelding vervangen nu het bestand example.c niet in het proces is, maar het bestand hello.c wel. Het is te zien dat de proces-ID hetzelfde is, of hello.c nu een procesafbeelding is of voorbeeld.c een procesafbeelding is, omdat het proces hetzelfde is en de procesafbeelding alleen wordt vervangen.
Dan hebben we nog iets om op te merken, namelijk de printf()-instructie nadat execv() niet is uitgevoerd. Dit komt omdat de besturing nooit teruggaat naar de oude procesafbeelding zodra deze door een nieuwe procesafbeelding wordt vervangen. De besturing keert pas terug naar de oproepfunctie als het vervangen van de procesafbeelding niet is gelukt. (De retourwaarde is in dit geval -1).
Verschil tussen fork() en exec() systeemaanroepen:
De systeemaanroep fork() wordt gebruikt om een exacte kopie van een lopend proces te maken en de gemaakte kopie is het onderliggende proces en het lopende proces is het bovenliggende proces. Terwijl de systeemaanroep exec() wordt gebruikt om een procesafbeelding te vervangen door een nieuwe procesafbeelding. Daarom is er geen concept van bovenliggende en onderliggende processen in de systeemaanroep exec().
In fork() systeemaanroep worden de bovenliggende en onderliggende processen tegelijkertijd uitgevoerd. Maar in exec() systeemaanroep, als de vervanging van de procesafbeelding succesvol is, keert het besturingselement niet terug naar waar de exec-functie werd aangeroepen, maar voert het het nieuwe proces uit. De besturing wordt alleen terug overgedragen als er een fout is.
Voorbeeld 2: Fork() en exec() systeemaanroepen combineren
Beschouw het volgende voorbeeld waarin we zowel fork() als exec() systeemaanroepen in hetzelfde programma hebben gebruikt:
voorbeeld.c
CODE:
#erbij betrekken#erbij betrekken
#erbij betrekken
inthoofd(intargc, char *argv[])
{
printf ('PID van voorbeeld.c = %dN',getpid());
pid_t p;
P=vork();
indien(P== -1)
{
printf ('Er is een fout opgetreden bij het aanroepen van fork()');
}
indien(P==0)
{
printf ('We zitten in het kindproces'N');
printf ('Hallo.c bellen vanuit kindproces'N');
char *argumenten[] = {'Hallo', 'C', 'Programmeren',NUL};
execv('./Hallo',argumenten);
}
anders
{
printf ('We zitten in het ouderproces');
}
opbrengst 0;
}
hallo.c:
CODE:
#erbij betrekken#erbij betrekken
#erbij betrekken
inthoofd(intargc, char *argv[])
{
printf ('We zijn in Hallo.cN');
printf ('PID van hallo.c = %dN',getpid());
opbrengst 0;
}
UITGANG:
PID van voorbeeld.c = 4790We zitten in het ouderproces
We zitten in het kinderproces
Bellen naar hello.c vanuit het onderliggende proces
We zijn in hallo.c
PID van hallo.c = 4791
In dit voorbeeld hebben we fork() system call gebruikt. Wanneer het onderliggende proces is gemaakt, wordt 0 toegewezen aan p en gaan we naar het onderliggende proces. Nu wordt het blok met instructies met if(p==0) uitgevoerd. Er wordt een bericht weergegeven en we hebben de systeemaanroep execv() gebruikt en de huidige afbeelding van het onderliggende proces, die example.c is, wordt vervangen door hello.c. Vóór execv() waren de onderliggende en bovenliggende processen hetzelfde.
Het is te zien dat de PID van example.c en hello.c nu anders is. Dit komt omdat example.c de bovenliggende procesafbeelding is en hello.c de onderliggende procesafbeelding is.