CWE Rule 88
Improper Neutralization of Argument Delimiters in a Command ('Argument Injection')
Since R2024a
Description
Rule Description
The product constructs a string for a command to be executed by a separate component in another control sphere, but it does not properly delimit the intended arguments, options, or switches within that command string.
Polyspace Implementation
The rule checker checks for these issues:
Execution of externally controlled command
Unsafe call to a system function
Examples
This issue occurs when commands are fully or partially constructed from externally controlled input.
Attackers can use the externally controlled input as operating system commands, or arguments to the application. An attacker could read or modify sensitive data can be read or modified, execute unintended code, or gain access to other aspects of the program.
Validate the inputs to allow only intended input values. For example, create a list of acceptable inputs and compare the input against this list.
By default, Polyspace® assumes that data from external sources are tainted. See Sources of Tainting in a Polyspace Analysis. To consider
any data that does not originate in the current scope of Polyspace analysis as
tainted, use the command line option -consider-analysis-perimeter-as-trust-boundary
.
#include "stdlib.h"
#include "stdio.h"
#include "string.h"
#include "unistd.h"
#include "dlfcn.h"
#include "limits.h"
#define MAX 128
void taintedexternalcmd(void)
{
char* usercmd;
fgets(usercmd,MAX,stdin);
char cmd[MAX] = "/usr/bin/cat ";
strcat(cmd, usercmd);
system(cmd);//Noncompliant
}
This example function calls a command from a user input without checking the command variable.
One possible correction is to use a switch
statement to run a
predefined command, using the user input as the switch variable. This corrected code also
replaces the call to system()
using calls to
execve()
, which is less unsafe.
#define _XOPEN_SOURCE
#define _GNU_SOURCE
#include "stdlib.h"
#include "stdio.h"
#include "string.h"
#include "unistd.h"
#include "dlfcn.h"
#include "limits.h"
enum {
SIZE10 = 10,
SIZE100 = 100,
SIZE128 = 128
};
enum { CMD0 = 1, CMD1, CMD2 };
void taintedexternalcmd(void)
{
int usercmd = strtol(getenv("cmd"), NULL, 10);
char *argv[3];
char cmd[SIZE128] = "/usr/bin/cat";
char arg[SIZE10];
switch(usercmd) {
case CMD0:
strcpy(arg, "*.c");
break;
case CMD1:
strcpy(arg, "*.h");
break;
case CMD2:
strcpy(arg, "*.cpp");
break;
default:
strcpy(arg, "*.c");
}
argv[0] = cmd;
argv[1] = arg;
argv[2] = NULL;
execve(cmd, argv, NULL);
// If execve returns, there was an error
perror("execve");
exit(EXIT_FAILURE);
}
This issue occurs when you use a function that invokes an implementation-defined command processor. These functions include:
The C standard
system()
function.The POSIX
popen()
function.The Windows®
_popen()
and_wpopen()
functions.
If the argument of a function that invokes a command processor is not sanitized, it can cause exploitable vulnerabilities. An attacker can execute arbitrary commands or read and modify data anywhere on the system.
Do not use a system
-family function to invoke a command processor.
Instead, use safer functions such as POSIX execve()
and WinAPI
CreateProcess()
.
# include <string.h>
# include <stdlib.h>
# include <stdio.h>
# include <unistd.h>
enum {
SIZE512=512,
SIZE3=3};
void func(char *arg)
{
char buf[SIZE512];
int retval=sprintf(buf, "/usr/bin/any_cmd %s", arg);
if (retval<=0 || retval>SIZE512){
/* Handle error */
abort();
}
/* Use of system() to pass any_cmd with
unsanitized argument to command processor */
if (system(buf) == -1) { //Noncompliant
/* Handle error */
}
}
In this example, system()
passes its argument to the host environment
for the command processor to execute. This code is vulnerable to an attack by
command-injection.
execve()
In the following code, the argument of any_cmd
is sanitized, and
then passed to execve()
for execution. exec
-family
functions are not vulnerable to command-injection attacks.
# include <string.h>
# include <stdlib.h>
# include <stdio.h>
# include <unistd.h>
enum {
SIZE512=512,
SIZE3=3};
void func(char *arg)
{
char *const args[SIZE3] = {"any_cmd", arg, NULL};
char *const env[] = {NULL};
/* Sanitize argument */
/* Use execve() to execute any_cmd. */
if (execve("/usr/bin/time", args, env) == -1) {
/* Handle error */
}
}
Check Information
Category: Others |
Version History
Introduced in R2024a
See Also
External Websites
MATLAB Command
You clicked a link that corresponds to this MATLAB command:
Run the command by entering it in the MATLAB Command Window. Web browsers do not support MATLAB commands.
Seleccione un país/idioma
Seleccione un país/idioma para obtener contenido traducido, si está disponible, y ver eventos y ofertas de productos y servicios locales. Según su ubicación geográfica, recomendamos que seleccione: .
También puede seleccionar uno de estos países/idiomas:
Cómo obtener el mejor rendimiento
Seleccione China (en idioma chino o inglés) para obtener el mejor rendimiento. Los sitios web de otros países no están optimizados para ser accedidos desde su ubicación geográfica.
América
- América Latina (Español)
- Canada (English)
- United States (English)
Europa
- Belgium (English)
- Denmark (English)
- Deutschland (Deutsch)
- España (Español)
- Finland (English)
- France (Français)
- Ireland (English)
- Italia (Italiano)
- Luxembourg (English)
- Netherlands (English)
- Norway (English)
- Österreich (Deutsch)
- Portugal (English)
- Sweden (English)
- Switzerland
- United Kingdom (English)