Gimp gdk_pixbuf_save не работает с AccessViolationException, но я не знаю почему?
Я использую код, который я нашел, чтобы растеризовать файлы SVG в файлы изображений. Ссылка http://www.experts-exchange.com/Programming/Languages/C_Sharp/Q_24547525.html. Мой код выглядит идентично, но я получаю нарушение прав доступа:
"Попытка чтения или записи защищенной памяти. Это часто указывает на то, что другая память повреждена".
Чтобы устранить очевидное: я использую файл svg из каталога Gimp, поэтому у меня есть известный хороший файл. Я использую e: в качестве сохранения в файл, чтобы библиотека не затрагивала какие-либо элементы безопасности / разрешений. Я запускаю Visual Studio с правами администратора в случае, если это проблема. У меня установлен 64-битный компьютер, и для Local.testsettings задано значение "Запускать тестирование в 64-процессорной системе на 64-битной машине"
Есть идеи?
Мой код:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Runtime.InteropServices;
using System.ComponentModel;
namespace TA.Rasterizer
{
/// <summary>
/// Takes SVG image file type and converts to PNG image file type.
/// Gimp must be installed. Gimp dll location must be in the web.config
/// file key "ThreatAnalyzer-DllDirectoryForGimp".
/// <summary>
/// http://msdn.microsoft.com/en-us/magazine/cc164123.aspx
/// http://www.experts-exchange.com/Programming/Languages/C_Sharp/Q_24547525.html
/// </summary> /// </summary>
public static class Rasterizer
{
// C:\Windows\System32
[DllImport("kernel32.dll", SetLastError = true)]
static extern bool SetDllDirectory(string pathname);
[DllImport("libgobject-2.0-0.dll", SetLastError = true)]
static extern void g_type_init();
[DllImport("librsvg-2-2.dll", SetLastError = true)]
static extern IntPtr rsvg_pixbuf_from_file_at_size(string file_name, int width, int height, out IntPtr error);
[DllImport("libgdk_pixbuf-2.0-0.dll", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Auto)]
static extern bool gdk_pixbuf_save(IntPtr pixbuf, string filename, string type, out IntPtr error, __arglist);
//static extern bool gdk_pixbuf_save_to_buffer(
/// <summary>
/// Get location of Gimp installed Dlls from web.config appsettings key/value.
/// Key name is found in this.webConfigFileAppSettingsGimp
/// </summary>
/// <returns></returns>
private static string GetGimpDllLocationFromConfigurationFile(bool webapp, string nonDefaultConfigFileName)
{
// connect to config file
Configuration rasterizerConfiguration = new TA.Rasterizer.Configuration(webapp, nonDefaultConfigFileName);
// grab config setting
return rasterizerConfiguration.Get("ThreatAnalyzer-DllDirectoryForGimp");
}
/// <summary>
/// http://msdn.microsoft.com/en-us/magazine/cc164123.aspx
/// You must check that file exists after call in order to check success.
/// </summary>
/// <param name="inputFileName">SVG file to be converted</param>
/// <param name="outputFileName">PNG file as final product</param>
/// <returns></returns>
public static void SvgToPng(string inputFileName, string outputFileName, bool webapp, string nonDefaultConfigFileName)
{
string gimpDllLocation = GetGimpDllLocationFromConfigurationFile(webapp, nonDefaultConfigFileName);
bool rasterizeSuccess = ExternRaster.RasterizeSvg(inputFileName, outputFileName, gimpDllLocation);
if (!rasterizeSuccess)
{
Int32 err = System.Runtime.InteropServices.Marshal.GetLastWin32Error();
throw new Win32Exception(err);
}
}
/// <summary>
/// http://msdn.microsoft.com/en-us/magazine/cc164123.aspx
/// http://www.experts-exchange.com/Programming/Languages/C_Sharp/Q_24547525.html
/// </summary>
internal sealed class ExternRaster
{
public static bool RasterizeSvg(string inputFileName, string outputFileName, string gimpDllLocation)
{
bool callSuccessful = SetDllDirectory(gimpDllLocation);
if (!callSuccessful)
{
throw new Exception("Could not set DLL directory");
}
g_type_init();
IntPtr error1;
IntPtr result = rsvg_pixbuf_from_file_at_size(@"E:\green.svg", -1, -1, out error1);
if (error1 != IntPtr.Zero)
{
throw new Exception(Marshal.ReadInt32(error1).ToString());
}
IntPtr error2;
callSuccessful = gdk_pixbuf_save(result, @"E:\green.jpg", "jpg", out error2, __arglist(null));
if (!callSuccessful)
{
throw new Exception(error2.ToInt32().ToString());
}
return true;
}
}
}
}
Я получаю доступ к коду из модульного теста MSTest:
/// <summary>
///A test for RasterizeSvg
///</summary>
[TestMethod()]
public void RasterizeSvgTest()
{
// arrange
string svgFileNameInAppDirectory = "1kijvsrewxxr3mdvlwitjvyu-pie.svg";
string currentClassLibDirectory = Environment.CurrentDirectory.Replace(@"\bin\debug\","");
string pathToAppData = Setup.GetDataPath();
string inputFileName = Path.Combine(pathToAppData, svgFileNameInAppDirectory);
string outputFileName = Path.Combine(pathToAppData, svgFileNameInAppDirectory.Replace("svg","png"));
bool webapp = false;
string nonDefaultConfigFileName = string.Empty;
// act
Rasterizer.SvgToPng(inputFileName, outputFileName, webapp, nonDefaultConfigFileName);
// assert
bool fileExists = File.Exists(outputFileName);
Assert.IsTrue(fileExists);
// cleanup
if (fileExists)
{
File.Delete(outputFileName);
}
}
1 ответ
Документация дляrsvg_pixbuf_from_file_at_size
Говорит, что
Если произошла ошибка, устанавливается ошибка и возвращается NULL.
Для меня это может означать, что в случае успеха ваш error1
переменная не установлена в допустимый указатель. Вы должны проверить возвращаемое значение result
вместо.
Я думаю, что то, что @GojiraDemonstah сказал о "JPG" против "JPEG" также важно.
Дополнительно, rsvg_pixbuf_from_file_at_size
выделяет новый GdkPixbuf
и у вас нет кода, чтобы освободить его. Для этого вы также должны иметь
[DllImport("libgobject-2.0-0.dll", SetLastError = true)]
static extern void g_object_unref(IntPtr obj);
в вашем импорте и
g_object_unref(result);
после звонка gdk_pixbuf_save
чтобы избежать утечки памяти.