Blame SOURCES/gpgme-1.13.1-fix-resource-leaks.patch

d2ead4
diff --git a/lang/cpp/src/context.cpp b/lang/cpp/src/context.cpp
d2ead4
index c0a1dc21..e991dbc8 100644
d2ead4
--- a/lang/cpp/src/context.cpp
d2ead4
+++ b/lang/cpp/src/context.cpp
d2ead4
@@ -223,6 +223,7 @@ Context *Context::createForProtocol(Protocol proto)
d2ead4
         }
d2ead4
         break;
d2ead4
     default:
d2ead4
+        gpgme_release(ctx);
d2ead4
         return nullptr;
d2ead4
     }
d2ead4
 
d2ead4
@@ -273,6 +274,7 @@ std::unique_ptr<Context> Context::createForEngine(Engine eng, Error *error)
d2ead4
         }
d2ead4
         break;
d2ead4
     default:
d2ead4
+        gpgme_release(ctx);
d2ead4
         if (error) {
d2ead4
             *error = Error::fromCode(GPG_ERR_INV_ARG);
d2ead4
         }
d2ead4
diff --git a/lang/cpp/src/data.cpp b/lang/cpp/src/data.cpp
d2ead4
index 7a93cbc2..d08a29db 100644
d2ead4
--- a/lang/cpp/src/data.cpp
d2ead4
+++ b/lang/cpp/src/data.cpp
d2ead4
@@ -249,6 +249,7 @@ std::vector<GpgME::Key> GpgME::Data::toKeys(Protocol proto) const
d2ead4
     }
d2ead4
 
d2ead4
     if (gpgme_op_keylist_from_data_start (ctx->impl()->ctx, d->data, 0)) {
d2ead4
+        delete ctx;
d2ead4
         return ret;
d2ead4
     }
d2ead4
 
d2ead4
diff --git a/lang/python/gpgme.i b/lang/python/gpgme.i
d2ead4
index 87371af8..64b88d54 100644
d2ead4
--- a/lang/python/gpgme.i
d2ead4
+++ b/lang/python/gpgme.i
d2ead4
@@ -53,7 +53,7 @@
d2ead4
     {
d2ead4
       encodedInput = PyUnicode_AsUTF8String($input);
d2ead4
       if (encodedInput == NULL)
d2ead4
-        return NULL;
d2ead4
+        SWIG_fail;
d2ead4
       $1 = PyBytes_AsString(encodedInput);
d2ead4
     }
d2ead4
   else if (PyBytes_Check($input))
d2ead4
@@ -62,22 +62,25 @@
d2ead4
     PyErr_Format(PyExc_TypeError,
d2ead4
                  "arg %d: expected str, bytes, or None, got %s",
d2ead4
 		 $argnum, $input->ob_type->tp_name);
d2ead4
-    return NULL;
d2ead4
+    SWIG_fail;
d2ead4
   }
d2ead4
 }
d2ead4
 %typemap(freearg) const char * {
d2ead4
   Py_XDECREF(encodedInput$argnum);
d2ead4
 }
d2ead4
 
d2ead4
+%typemap(arginit) const char *[] {
d2ead4
+  $1 = NULL;
d2ead4
+}
d2ead4
+
d2ead4
 /* Likewise for a list of strings.  */
d2ead4
-%typemap(in) const char *[] (void *vector = NULL,
d2ead4
-                             size_t size,
d2ead4
+%typemap(in) const char *[] (size_t size,
d2ead4
                              PyObject **pyVector = NULL) {
d2ead4
   /* Check if is a list */
d2ead4
   if (PyList_Check($input)) {
d2ead4
     size_t i, j;
d2ead4
     size = PyList_Size($input);
d2ead4
-    $1 = (char **) (vector = malloc((size+1) * sizeof(char *)));
d2ead4
+    $1 = (char **) malloc((size+1) * sizeof(char *));
d2ead4
     pyVector = calloc(sizeof *pyVector, size);
d2ead4
 
d2ead4
     for (i = 0; i < size; i++) {
d2ead4
@@ -86,12 +89,7 @@
d2ead4
         {
d2ead4
           pyVector[i] = PyUnicode_AsUTF8String(o);
d2ead4
           if (pyVector[i] == NULL)
d2ead4
-            {
d2ead4
-              free(vector);
d2ead4
-              for (j = 0; j < i; j++)
d2ead4
-                Py_XDECREF(pyVector[j]);
d2ead4
-              return NULL;
d2ead4
-            }
d2ead4
+            SWIG_fail;
d2ead4
           $1[i] = PyBytes_AsString(pyVector[i]);
d2ead4
         }
d2ead4
       else if (PyString_Check(o))
d2ead4
@@ -101,8 +99,7 @@
d2ead4
                      "arg %d: list must contain only str or bytes, got %s "
d2ead4
                      "at position %d",
d2ead4
                      $argnum, o->ob_type->tp_name, i);
d2ead4
-	free($1);
d2ead4
-	return NULL;
d2ead4
+	SWIG_fail;
d2ead4
       }
d2ead4
     }
d2ead4
     $1[i] = NULL;
d2ead4
@@ -110,14 +107,17 @@
d2ead4
     PyErr_Format(PyExc_TypeError,
d2ead4
                  "arg %d: expected a list of str or bytes, got %s",
d2ead4
                  $argnum, $input->ob_type->tp_name);
d2ead4
-    return NULL;
d2ead4
+    SWIG_fail;
d2ead4
   }
d2ead4
 }
d2ead4
 %typemap(freearg) const char *[] {
d2ead4
-  size_t i;
d2ead4
-  free(vector$argnum);
d2ead4
-  for (i = 0; i < size$argnum; i++)
d2ead4
-    Py_XDECREF(pyVector$argnum[i]);
d2ead4
+  if (pyVector$argnum) {
d2ead4
+    size_t i;
d2ead4
+    for (i = 0; i < size$argnum; i++)
d2ead4
+      Py_XDECREF(pyVector$argnum[i]);
d2ead4
+    free(pyVector$argnum);
d2ead4
+  }
d2ead4
+  if ($1) free($1);
d2ead4
 }
d2ead4
 
d2ead4
 /* Release returned buffers as necessary.  */
d2ead4
@@ -135,7 +135,7 @@
d2ead4
   if (!PySequence_Check($input)) {
d2ead4
     PyErr_Format(PyExc_ValueError, "arg %d: Expected a list of gpgme_key_t",
d2ead4
 		 $argnum);
d2ead4
-    return NULL;
d2ead4
+    SWIG_fail;
d2ead4
   }
d2ead4
   if((numb = PySequence_Length($input)) != 0) {
d2ead4
     $1 = (gpgme_key_t*)malloc((numb+1)*sizeof(gpgme_key_t));
d2ead4
@@ -152,8 +152,7 @@
d2ead4
                      "arg %d: list must contain only gpgme_key_ts, got %s "
d2ead4
                      "at position %d",
d2ead4
                      $argnum, pypointer->ob_type->tp_name, i);
d2ead4
-        free($1);
d2ead4
-	return NULL;
d2ead4
+	SWIG_fail;
d2ead4
       }
d2ead4
       Py_DECREF(pypointer);
d2ead4
     }
d2ead4
@@ -179,7 +178,7 @@
d2ead4
     pypointer = _gpg_obj2gpgme_data_t($input, $argnum, &wrapper,
d2ead4
                                        &bytesio, &view);
d2ead4
     if (pypointer == NULL)
d2ead4
-      return NULL;
d2ead4
+      SWIG_fail;
d2ead4
     have_view = !! view.obj;
d2ead4
 
d2ead4
     /* input = $input, 1 = $1, 1_descriptor = $1_descriptor */
d2ead4
@@ -189,7 +188,7 @@
d2ead4
     if ((SWIG_ConvertPtr(pypointer,(void **) &$1, $1_descriptor,
d2ead4
          SWIG_POINTER_EXCEPTION | $disown )) == -1) {
d2ead4
       Py_DECREF(pypointer);
d2ead4
-      return NULL;
d2ead4
+      SWIG_fail;
d2ead4
     }
d2ead4
     Py_DECREF(pypointer);
d2ead4
   }
d2ead4
@@ -346,6 +345,11 @@
d2ead4
     PyErr_SetString(PyExc_TypeError, "Numeric argument expected");
d2ead4
 }
d2ead4
 
d2ead4
+%typemap(arginit) (void *buffer, size_t size), (char *buf, size_t buflen) {
d2ead4
+  $1 = NULL;
d2ead4
+  $2 = 0;
d2ead4
+}
d2ead4
+
d2ead4
 /* Those are for gpgme_data_read() and gpgme_strerror_r().  */
d2ead4
 %typemap(in) (void *buffer, size_t size), (char *buf, size_t buflen) {
d2ead4
   {
d2ead4
@@ -359,12 +363,12 @@
d2ead4
     else
d2ead4
       {
d2ead4
         PyErr_SetString(PyExc_TypeError, "Numeric argument expected");
d2ead4
-        return NULL;
d2ead4
+        SWIG_fail;
d2ead4
       }
d2ead4
 
d2ead4
     if (tmp$argnum < 0) {
d2ead4
       PyErr_SetString(PyExc_ValueError, "Positive integer expected");
d2ead4
-      return NULL;
d2ead4
+      SWIG_fail;
d2ead4
     }
d2ead4
     $2 = (size_t) tmp$argnum;
d2ead4
     $1 = ($1_ltype) malloc($2+1);
d2ead4
@@ -373,11 +377,11 @@
d2ead4
 %typemap(argout) (void *buffer, size_t size), (char *buf, size_t buflen) {
d2ead4
   Py_XDECREF($result);   /* Blow away any previous result */
d2ead4
   if (result < 0) {      /* Check for I/O error */
d2ead4
-    free($1);
d2ead4
+    if ($1) free($1);
d2ead4
     return PyErr_SetFromErrno(PyExc_RuntimeError);
d2ead4
   }
d2ead4
   $result = PyBytes_FromStringAndSize($1,result);
d2ead4
-  free($1);
d2ead4
+  if ($1) free($1);
d2ead4
 }
d2ead4
 
d2ead4
 /* For gpgme_data_write, but should be universal.  */
d2ead4
@@ -390,11 +394,11 @@
d2ead4
     {
d2ead4
       encodedInput = PyUnicode_AsUTF8String($input);
d2ead4
       if (encodedInput == NULL)
d2ead4
-        return NULL;
d2ead4
+        SWIG_fail;
d2ead4
       if (PyBytes_AsStringAndSize(encodedInput, (char **) &$1, &ssize) == -1)
d2ead4
         {
d2ead4
           Py_DECREF(encodedInput);
d2ead4
-          return NULL;
d2ead4
+          SWIG_fail;
d2ead4
         }
d2ead4
     }
d2ead4
   else if (PyBytes_Check($input))
d2ead4
@@ -403,7 +407,7 @@
d2ead4
     PyErr_Format(PyExc_TypeError,
d2ead4
                  "arg %d: expected str, bytes, or None, got %s",
d2ead4
 		 $argnum, $input->ob_type->tp_name);
d2ead4
-    return NULL;
d2ead4
+    SWIG_fail;
d2ead4
   }
d2ead4
 
d2ead4
   if (! $1)
d2ead4
@@ -432,8 +436,17 @@
d2ead4
     size++;
d2ead4
   }
d2ead4
   $result = PyList_New(size);
d2ead4
+  if ($result == NULL)
d2ead4
+    SWIG_fail;
d2ead4
   for (i=0,curr=$1; i<size; i++,curr=curr->next) {
d2ead4
     PyObject *o = SWIG_NewPointerObj(SWIG_as_voidptr(curr), $1_descriptor, %newpointer_flags);
d2ead4
+    if (o == NULL) {
d2ead4
+      int j;
d2ead4
+      for (j = 0; j < i; j++)
d2ead4
+        Py_XDECREF(PyList_GetItem($result, j));
d2ead4
+      Py_DECREF($result);
d2ead4
+      SWIG_fail;
d2ead4
+    }
d2ead4
     PyList_SetItem($result, i, o);
d2ead4
   }
d2ead4
 }
d2ead4
@@ -446,6 +459,8 @@
d2ead4
   PyObject *fragile;
d2ead4
   fragile = SWIG_NewPointerObj(SWIG_as_voidptr($1), $1_descriptor,
d2ead4
                                %newpointer_flags);
d2ead4
+  if (fragile == NULL)
d2ead4
+    SWIG_fail;
d2ead4
   $result = _gpg_wrap_result(fragile, name);
d2ead4
   Py_DECREF(fragile);
d2ead4
 }
d2ead4
@@ -469,22 +484,28 @@ wrapresult(gpgme_vfs_mount_result_t, "VFSMountResult")
d2ead4
   }
d2ead4
   $result = PyList_New(size);
d2ead4
   if ($result == NULL)
d2ead4
-    return NULL;	/* raise */
d2ead4
+    SWIG_fail;	/* raise */
d2ead4
   for (i=0,curr=$1; i<size; i++,curr=curr->next) {
d2ead4
     PyObject *fragile, *o;
d2ead4
     fragile = SWIG_NewPointerObj(SWIG_as_voidptr(curr), $1_descriptor,
d2ead4
                                  %newpointer_flags);
d2ead4
     if (fragile == NULL)
d2ead4
       {
d2ead4
+        int j;
d2ead4
+        for (j = 0; j < i; j++)
d2ead4
+          Py_XDECREF(PyList_GetItem($result, j));
d2ead4
         Py_DECREF($result);
d2ead4
-        return NULL;	/* raise */
d2ead4
+        SWIG_fail;	/* raise */
d2ead4
       }
d2ead4
     o = _gpg_wrap_result(fragile, "EngineInfo");
d2ead4
     Py_DECREF(fragile);
d2ead4
     if (o == NULL)
d2ead4
       {
d2ead4
+        int j;
d2ead4
+        for (j = 0; j < i; j++)
d2ead4
+          Py_XDECREF(PyList_GetItem($result, j));
d2ead4
         Py_DECREF($result);
d2ead4
-        return NULL;	/* raise */
d2ead4
+        SWIG_fail;	/* raise */
d2ead4
       }
d2ead4
     PyList_SetItem($result, i, o);
d2ead4
   }
d2ead4
diff --git a/src/engine-gpg.c b/src/engine-gpg.c
d2ead4
index dc2d9455..a1e40fe2 100644
d2ead4
--- a/src/engine-gpg.c
d2ead4
+++ b/src/engine-gpg.c
d2ead4
@@ -1122,6 +1122,7 @@ build_argv (engine_gpg_t gpg, const char *pgmname)
d2ead4
                    to avoid and given that we reach this here only
d2ead4
                    after a malloc failure for a small object, it is
d2ead4
                    probably better not to do anything.  */
d2ead4
+		free_argv (argv);
d2ead4
 		return gpg_error (GPG_ERR_GENERAL);
d2ead4
 	      }
d2ead4
 	    /* If the data_type is FD, we have to do a dup2 here.  */
d2ead4
@@ -1239,6 +1240,9 @@ read_status (engine_gpg_t gpg)
d2ead4
       buffer = realloc (buffer, bufsize);
d2ead4
       if (!buffer)
d2ead4
 	return gpg_error_from_syserror ();
d2ead4
+      /* Update buffer and bufsize here to prevent memory leaks.  */
d2ead4
+      gpg->status.buffer = buffer;
d2ead4
+      gpg->status.bufsize = bufsize;
d2ead4
     }
d2ead4
 
d2ead4
   nread = _gpgme_io_read (gpg->status.fd[0],
d2ead4
@@ -1351,8 +1355,6 @@ read_status (engine_gpg_t gpg)
d2ead4
     }
d2ead4
 
d2ead4
   /* Update the gpg object.  */
d2ead4
-  gpg->status.bufsize = bufsize;
d2ead4
-  gpg->status.buffer = buffer;
d2ead4
   gpg->status.readpos = readpos;
d2ead4
   return 0;
d2ead4
 }
d2ead4
@@ -1392,6 +1394,9 @@ read_colon_line (engine_gpg_t gpg)
d2ead4
       buffer = realloc (buffer, bufsize);
d2ead4
       if (!buffer)
d2ead4
 	return gpg_error_from_syserror ();
d2ead4
+      /* Prevent memory leaks.  */
d2ead4
+      gpg->colon.bufsize = bufsize;
d2ead4
+      gpg->colon.buffer  = buffer;
d2ead4
     }
d2ead4
 
d2ead4
   nread = _gpgme_io_read (gpg->colon.fd[0], buffer+readpos, bufsize-readpos);
d2ead4
@@ -1471,8 +1476,6 @@ read_colon_line (engine_gpg_t gpg)
d2ead4
     }
d2ead4
 
d2ead4
   /* Update the gpg object.  */
d2ead4
-  gpg->colon.bufsize = bufsize;
d2ead4
-  gpg->colon.buffer  = buffer;
d2ead4
   gpg->colon.readpos = readpos;
d2ead4
   return 0;
d2ead4
 }
d2ead4
diff --git a/src/engine-gpgsm.c b/src/engine-gpgsm.c
d2ead4
index ae5d8ef1..1f603f19 100644
d2ead4
--- a/src/engine-gpgsm.c
d2ead4
+++ b/src/engine-gpgsm.c
d2ead4
@@ -1533,8 +1533,10 @@ gpgsm_export (void *engine, const char *pattern, gpgme_export_mode_t mode,
d2ead4
   gpgsm->output_cb.data = keydata;
d2ead4
   err = gpgsm_set_fd (gpgsm, OUTPUT_FD, use_armor ? "--armor"
d2ead4
 		      : map_data_enc (gpgsm->output_cb.data));
d2ead4
-  if (err)
d2ead4
+  if (err) {
d2ead4
+    free (cmd);
d2ead4
     return err;
d2ead4
+  }
d2ead4
   gpgsm_clear_fd (gpgsm, INPUT_FD);
d2ead4
   gpgsm_clear_fd (gpgsm, MESSAGE_FD);
d2ead4
   gpgsm->inline_data = NULL;
d2ead4
@@ -1634,8 +1636,10 @@ gpgsm_export_ext (void *engine, const char *pattern[], gpgme_export_mode_t mode,
d2ead4
   gpgsm->output_cb.data = keydata;
d2ead4
   err = gpgsm_set_fd (gpgsm, OUTPUT_FD, use_armor ? "--armor"
d2ead4
 		      : map_data_enc (gpgsm->output_cb.data));
d2ead4
-  if (err)
d2ead4
+  if (err) {
d2ead4
+    free (line);
d2ead4
     return err;
d2ead4
+  }
d2ead4
   gpgsm_clear_fd (gpgsm, INPUT_FD);
d2ead4
   gpgsm_clear_fd (gpgsm, MESSAGE_FD);
d2ead4
   gpgsm->inline_data = NULL;
d2ead4
diff --git a/src/engine.c b/src/engine.c
d2ead4
index 05979c15..7e2e3800 100644
d2ead4
--- a/src/engine.c
d2ead4
+++ b/src/engine.c
d2ead4
@@ -460,7 +460,8 @@ _gpgme_set_engine_info (gpgme_engine_info_t info, gpgme_protocol_t proto,
d2ead4
       if (!new_version)
d2ead4
         {
d2ead4
           free (new_file_name);
d2ead4
-          free (new_home_dir);
d2ead4
+          if (new_home_dir)
d2ead4
+            free (new_home_dir);
d2ead4
           return gpg_error_from_syserror ();
d2ead4
         }
d2ead4
     }
d2ead4
diff --git a/src/gpgme-tool.c b/src/gpgme-tool.c
d2ead4
index 7a0bfcb3..f4ddd83b 100644
d2ead4
--- a/src/gpgme-tool.c
d2ead4
+++ b/src/gpgme-tool.c
d2ead4
@@ -1189,6 +1189,7 @@ gt_get_key (gpgme_tool_t gt, const char *pattern, gpgme_key_t *r_key)
d2ead4
                  similar hack to sort out such duplicates but it can't
d2ead4
                  do that while listing keys.  */
d2ead4
               gpgme_key_unref (key);
d2ead4
+              key = NULL;
d2ead4
               goto try_next_key;
d2ead4
             }
d2ead4
 	  if (!err)
d2ead4
diff --git a/src/keylist.c b/src/keylist.c
d2ead4
index cdb115fd..b7cbf3c3 100644
d2ead4
--- a/src/keylist.c
d2ead4
+++ b/src/keylist.c
d2ead4
@@ -1321,6 +1321,7 @@ gpgme_get_key (gpgme_ctx_t ctx, const char *fpr, gpgme_key_t *r_key,
d2ead4
                  similar hack to sort out such duplicates but it can't
d2ead4
                  do that while listing keys.  */
d2ead4
               gpgme_key_unref (key);
d2ead4
+              key = NULL;
d2ead4
               goto try_next_key;
d2ead4
             }
d2ead4
 	  if (!err)