diff options
author | maruel@chromium.org <maruel@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-08-26 18:57:22 +0000 |
---|---|---|
committer | maruel@chromium.org <maruel@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2009-08-26 18:57:22 +0000 |
commit | ef6edc4c1060c7da59fc550663ffaa9c66baeca8 (patch) | |
tree | 982b79ec29991fbae6c2b90dad869bac4ed1e22c /printing | |
parent | a0d670a4fe0fdaf5e404e5b7e7a72aa3d00ed472 (diff) | |
download | chromium_src-ef6edc4c1060c7da59fc550663ffaa9c66baeca8.zip chromium_src-ef6edc4c1060c7da59fc550663ffaa9c66baeca8.tar.gz chromium_src-ef6edc4c1060c7da59fc550663ffaa9c66baeca8.tar.bz2 |
Reverting 24474.
Caused a memory leak (detected by valgrind)
Review URL: http://codereview.chromium.org/174547
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@24489 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'printing')
-rw-r--r-- | printing/pdf_ps_metafile_linux.cc | 215 | ||||
-rw-r--r-- | printing/pdf_ps_metafile_linux.h | 55 | ||||
-rw-r--r-- | printing/pdf_ps_metafile_linux_unittest.cc | 48 |
3 files changed, 130 insertions, 188 deletions
diff --git a/printing/pdf_ps_metafile_linux.cc b/printing/pdf_ps_metafile_linux.cc index 7f3027b..83c17a9 100644 --- a/printing/pdf_ps_metafile_linux.cc +++ b/printing/pdf_ps_metafile_linux.cc @@ -6,77 +6,18 @@ #include <stdio.h> -#include <cairo.h> #include <cairo-pdf.h> #include <cairo-ps.h> #include "base/file_util.h" #include "base/logging.h" -namespace { - -// Tests if |surface| is valid. -bool IsSurfaceValid(cairo_surface_t* surface) { - return cairo_surface_status(surface) == CAIRO_STATUS_SUCCESS; -} - -// Tests if |context| is valid. -bool IsContextValid(cairo_t* context) { - return cairo_status(context) == CAIRO_STATUS_SUCCESS; -} - -// Destroys and resets |surface|. -void CleanUpSurface(cairo_surface_t** surface) { - if (*surface) { - cairo_surface_destroy(*surface); - *surface = NULL; - } -} - -// Destroys and resets |context|. -void CleanUpContext(cairo_t** context) { - if (*context) { - cairo_destroy(*context); - *context = NULL; - } -} - -// Callback function for Cairo to write PDF/PS stream. -// |dst_buffer| is actually a pointer of type `std::string*`. -cairo_status_t WriteCairoStream(void* dst_buffer, - const unsigned char* src_data, - unsigned int src_data_length) { - DCHECK(dst_buffer); - DCHECK(src_data); - DCHECK(src_data_length > 0); - - std::string* buffer = reinterpret_cast<std::string*>(dst_buffer); - buffer->append(reinterpret_cast<const char*>(src_data), src_data_length); - - return CAIRO_STATUS_SUCCESS; -} - -} // namespace - namespace printing { PdfPsMetafile::PdfPsMetafile(const FileFormat& format) : format_(format), surface_(NULL), context_(NULL), page_surface_(NULL), page_context_(NULL) { -} - -PdfPsMetafile::~PdfPsMetafile() { - // Releases resources if we forgot to do so. - CleanUp(); -} - -bool PdfPsMetafile::Init() { - // We need to check at least these two members to ensure Init() has not been - // called before. Passing these two checks also implies that surface_, - // page_surface_, and page_context_ are NULL, and current_page_ is empty. - DCHECK(!context_); - DCHECK(all_pages_.empty()); // Create an 1 by 1 Cairo surface for entire PDF/PS file. // The size for each page will be overwritten later in StartPage(). @@ -95,53 +36,51 @@ bool PdfPsMetafile::Init() { default: NOTREACHED(); - return false; + return; } // Cairo always returns a valid pointer. // Hence, we have to check if it points to a "nil" object. if (!IsSurfaceValid(surface_)) { DLOG(ERROR) << "Cannot create Cairo surface for PdfPsMetafile!"; - CleanUpSurface(&surface_); - return false; + cairo_surface_destroy(surface_); + return; } // Create a context. context_ = cairo_create(surface_); if (!IsContextValid(context_)) { DLOG(ERROR) << "Cannot create Cairo context for PdfPsMetafile!"; - CleanUpContext(&context_); - CleanUpSurface(&surface_); - return false; + cairo_destroy(context_); + cairo_surface_destroy(surface_); + return; } - return true; + // Since |context_| will keep a reference of |surface_|, we can decreace + // surface's reference count by one safely. + cairo_surface_destroy(surface_); } -bool PdfPsMetafile::Init(const void* src_buffer, size_t src_buffer_size) { - // We need to check at least these two members to ensure Init() has not been - // called before. Passing these two checks also implies that surface_, - // page_surface_, and page_context_ are NULL, and current_page_ is empty. - DCHECK(!context_); - DCHECK(all_pages_.empty()); - - if (src_buffer == NULL || src_buffer_size == 0) { - return false; - } +PdfPsMetafile::PdfPsMetafile(const FileFormat& format, + const void* src_buffer, + size_t src_buffer_size) + : format_(format), + surface_(NULL), context_(NULL), + page_surface_(NULL), page_context_(NULL) { all_pages_ = std::string(reinterpret_cast<const char*>(src_buffer), src_buffer_size); +} - return true; +PdfPsMetafile::~PdfPsMetafile() { + // Releases resources if we forgot to do so. + Close(); } -cairo_t* PdfPsMetafile::StartPage(double width_in_points, - double height_in_points) { +bool PdfPsMetafile::StartPage(double width_in_points, double height_in_points) { DCHECK(IsSurfaceValid(surface_)); DCHECK(IsContextValid(context_)); - // Passing this check implies page_surface_ is NULL, and current_page_ is - // empty. - DCHECK(!page_context_); + DCHECK(!page_surface_ && !page_context_); DCHECK_GT(width_in_points, 0.); DCHECK_GT(height_in_points, 0.); @@ -167,35 +106,38 @@ cairo_t* PdfPsMetafile::StartPage(double width_in_points, default: NOTREACHED(); - CleanUp(); - return NULL; + return false; } // Cairo always returns a valid pointer. // Hence, we have to check if it points to a "nil" object. if (!IsSurfaceValid(page_surface_)) { DLOG(ERROR) << "Cannot create Cairo surface for PdfPsMetafile!"; - CleanUp(); - return NULL; + cairo_surface_destroy(page_surface_); + return false; } // Create a context. page_context_ = cairo_create(page_surface_); if (!IsContextValid(page_context_)) { DLOG(ERROR) << "Cannot create Cairo context for PdfPsMetafile!"; - CleanUp(); - return NULL; + cairo_destroy(page_context_); + cairo_surface_destroy(page_surface_); + return false; } - return page_context_; + // Since |page_context_| will keep a reference of |page_surface_|, we can + // decreace surface's reference count by one safely. + cairo_surface_destroy(page_surface_); + + return true; } -bool PdfPsMetafile::FinishPage(float shrink) { +void PdfPsMetafile::FinishPage(float shrink) { DCHECK(IsSurfaceValid(surface_)); DCHECK(IsContextValid(context_)); DCHECK(IsSurfaceValid(page_surface_)); DCHECK(IsContextValid(page_context_)); - DCHECK(shrink > 0); // Flush all rendering for current page. cairo_surface_flush(page_surface_); @@ -226,15 +168,7 @@ bool PdfPsMetafile::FinishPage(float shrink) { default: NOTREACHED(); - CleanUp(); - return false; - } - - // Check if our surface is still valid after resizing. - if (!IsSurfaceValid(surface_)) { - DLOG(ERROR) << "Cannot resize Cairo surface for PdfPsMetafile!"; - CleanUp(); - return false; + return; } // Save context's states. @@ -280,68 +214,44 @@ bool PdfPsMetafile::FinishPage(float shrink) { cairo_surface_flush(surface_); // Destroy resoreces for current page. - CleanUpContext(&page_context_); + cairo_destroy(page_context_); + page_context_ = NULL; page_surface_ = NULL; current_page_.clear(); // Restore context's states. cairo_restore(context_); - - return true; } void PdfPsMetafile::Close() { - DCHECK(IsSurfaceValid(surface_)); - DCHECK(IsContextValid(context_)); - // Passing this check implies page_surface_ is NULL, and current_page_ is - // empty. - DCHECK(!page_context_); - - cairo_surface_finish(surface_); - DCHECK(!all_pages_.empty()); // Make sure we did get something. - - CleanUpContext(&context_); - CleanUpSurface(&surface_); + if (surface_ != NULL && IsSurfaceValid(surface_)) { + cairo_surface_finish(surface_); + surface_ = NULL; + } + if (context_ != NULL && IsContextValid(context_)) { + cairo_destroy(context_); + context_ = NULL; + } } unsigned int PdfPsMetafile::GetDataSize() const { - // We need to check at least these two members to ensure that either Init() - // has been called to initialize |all_pages_|, or metafile has been closed. - // Passing these two checks also implies that surface_, page_surface_, and - // page_context_ are NULL, and current_page_ is empty. - DCHECK(!context_); - DCHECK(!all_pages_.empty()); + DCHECK(!surface_ && !context_); return all_pages_.size(); } -bool PdfPsMetafile::GetData(void* dst_buffer, size_t dst_buffer_size) const { +void PdfPsMetafile::GetData(void* dst_buffer, size_t dst_buffer_size) const { + DCHECK(!surface_ && !context_); DCHECK(dst_buffer); - DCHECK(dst_buffer_size > 0); - // We need to check at least these two members to ensure that either Init() - // has been called to initialize |all_pages_|, or metafile has been closed. - // Passing these two checks also implies that surface_, page_surface_, and - // page_context_ are NULL, and current_page_ is empty. - DCHECK(!context_); - DCHECK(!all_pages_.empty()); size_t data_size = GetDataSize(); - if (dst_buffer_size > data_size) { - return false; - } - + if (data_size < dst_buffer_size) + dst_buffer_size = data_size; memcpy(dst_buffer, all_pages_.data(), dst_buffer_size); - - return true; } bool PdfPsMetafile::SaveTo(const FilePath& filename) const { - // We need to check at least these two members to ensure that either Init() - // has been called to initialize |all_pages_|, or metafile has been closed. - // Passing these two checks also implies that surface_, page_surface_, and - // page_context_ are NULL, and current_page_ is empty. - DCHECK(!context_); - DCHECK(!all_pages_.empty()); + DCHECK(!surface_ && !context_); const unsigned int data_size = GetDataSize(); const unsigned int bytes_written = @@ -354,13 +264,24 @@ bool PdfPsMetafile::SaveTo(const FilePath& filename) const { return true; } -void PdfPsMetafile::CleanUp() { - CleanUpContext(&context_); - CleanUpSurface(&surface_); - CleanUpContext(&page_context_); - CleanUpSurface(&page_surface_); - current_page_.clear(); - all_pages_.clear(); +cairo_status_t PdfPsMetafile::WriteCairoStream(void* dst_buffer, + const unsigned char* src_data, + unsigned int src_data_length) { + DCHECK(dst_buffer); + DCHECK(src_data); + + std::string* buffer = reinterpret_cast<std::string* >(dst_buffer); + buffer->append(reinterpret_cast<const char*>(src_data), src_data_length); + + return CAIRO_STATUS_SUCCESS; +} + +bool PdfPsMetafile::IsSurfaceValid(cairo_surface_t* surface) const { + return cairo_surface_status(surface) == CAIRO_STATUS_SUCCESS; +} + +bool PdfPsMetafile::IsContextValid(cairo_t* context) const { + return cairo_status(context) == CAIRO_STATUS_SUCCESS; } } // namespace printing diff --git a/printing/pdf_ps_metafile_linux.h b/printing/pdf_ps_metafile_linux.h index bb1ac75..a49f635 100644 --- a/printing/pdf_ps_metafile_linux.h +++ b/printing/pdf_ps_metafile_linux.h @@ -5,13 +5,12 @@ #ifndef PRINTING_PDF_PS_METAFILE_LINUX_H_ #define PRINTING_PDF_PS_METAFILE_LINUX_H_ +#include <cairo.h> + #include <string> #include "base/basictypes.h" -typedef struct _cairo_surface cairo_surface_t; -typedef struct _cairo cairo_t; - class FilePath; namespace printing { @@ -25,41 +24,34 @@ class PdfPsMetafile { PS, }; - // In the renderer process, callers should also call Init(void) to see if the - // metafile can obtain all necessary rendering resources. - // In the browser process, callers should also call Init(const void*, size_t) - // to initialize the buffer |all_pages_| to use SaveTo(). + // The constructor we should use in the renderer process. explicit PdfPsMetafile(const FileFormat& format); - ~PdfPsMetafile(); - - // Initializes to a fresh new metafile. Returns true on success. - // Note: Only call in the renderer to allocate rendering resources. - bool Init(); - - // Initializes a copy of metafile from PDF/PS stream data. - // Returns true on success. + // The constructor we should use in the browser process. // |src_buffer| should point to the shared memory which stores PDF/PS // contents generated in the renderer. - // Note: Only call in the browser to initialize |all_pages_|. - bool Init(const void* src_buffer, size_t src_buffer_size); + PdfPsMetafile(const FileFormat& format, + const void* src_buffer, + size_t src_buffer_size); - FileFormat GetFileFormat() const { return format_; } + ~PdfPsMetafile(); + + FileFormat GetFileFormat() { return format_; } // Prepares a new cairo surface/context for rendering a new page. - // The unit is in point (=1/72 in). - // Returns NULL when failed. - cairo_t* StartPage(double width, double height); + bool StartPage(double width, double height); // The unit is pt (=1/72 in). + + // Returns the Cairo context for rendering current page. + cairo_t* GetPageContext() const { return page_context_; } // Destroys the surface and the context used in rendering current page. // The results of current page will be appended into buffer |all_pages_|. - // Returns true on success // TODO(myhuang): I plan to also do page setup here (margins, the header // and the footer). At this moment, only pre-defined margins for US letter // paper are hard-coded here. // |shrink| decides the scaling factor to fit raw printing results into // printable area. - bool FinishPage(float shrink); + void FinishPage(float shrink); // Closes resulting PDF/PS file. No further rendering is allowed. void Close(); @@ -70,8 +62,7 @@ class PdfPsMetafile { // Copies PDF/PS contents stored in buffer |all_pages_| into |dst_buffer|. // This function should ONLY be called after PDF/PS file is closed. - // Returns true only when success. - bool GetData(void* dst_buffer, size_t dst_buffer_size) const; + void GetData(void* dst_buffer, size_t dst_buffer_size) const; // Saves PDF/PS contents stored in buffer |all_pages_| into |filename| on // the disk. @@ -79,8 +70,17 @@ class PdfPsMetafile { bool SaveTo(const FilePath& filename) const; private: - // Cleans up all resources. - void CleanUp(); + // Callback function for Cairo to write PDF/PS stream. + // |dst_buffer| is actually a pointer of type `std::string*`. + static cairo_status_t WriteCairoStream(void* dst_buffer, + const unsigned char* src_data, + unsigned int src_data_length); + + // Convenient function to test if |surface| is valid. + bool IsSurfaceValid(cairo_surface_t* surface) const; + + // Convenient function to test if |context| is valid. + bool IsContextValid(cairo_t* context) const; FileFormat format_; @@ -104,3 +104,4 @@ class PdfPsMetafile { } // namespace printing #endif // PRINTING_PDF_PS_METAFILE_LINUX_H_ + diff --git a/printing/pdf_ps_metafile_linux_unittest.cc b/printing/pdf_ps_metafile_linux_unittest.cc index 3ef0822..8728ba3 100644 --- a/printing/pdf_ps_metafile_linux_unittest.cc +++ b/printing/pdf_ps_metafile_linux_unittest.cc @@ -15,22 +15,31 @@ typedef struct _cairo cairo_t; TEST(PdfTest, Basic) { // Tests in-renderer constructor. printing::PdfPsMetafile pdf(printing::PdfPsMetafile::PDF); - EXPECT_TRUE(pdf.Init()); + cairo_t* context = pdf.GetPageContext(); + EXPECT_TRUE(context == NULL); // Renders page 1. - cairo_t* context = pdf.StartPage(72, 72); + EXPECT_TRUE(pdf.StartPage(72, 72)); + context = pdf.GetPageContext(); EXPECT_TRUE(context != NULL); // In theory, we should use Cairo to draw something on |context|. - EXPECT_TRUE(pdf.FinishPage(1.5)); + pdf.FinishPage(1.5); + context = pdf.GetPageContext(); + EXPECT_TRUE(context == NULL); // Renders page 2. - context = pdf.StartPage(64, 64); + EXPECT_TRUE(pdf.StartPage(64, 64)); + context = pdf.GetPageContext(); EXPECT_TRUE(context != NULL); // In theory, we should use Cairo to draw something on |context|. - EXPECT_TRUE(pdf.FinishPage(0.5)); + pdf.FinishPage(0.5); + context = pdf.GetPageContext(); + EXPECT_TRUE(context == NULL); // Closes the file. pdf.Close(); + context = pdf.GetPageContext(); + EXPECT_TRUE(context == NULL); // Checks data size. unsigned int size = pdf.GetDataSize(); @@ -41,8 +50,9 @@ TEST(PdfTest, Basic) { pdf.GetData(&buffer.front(), size); // Tests another constructor. - printing::PdfPsMetafile pdf2(printing::PdfPsMetafile::PDF); - EXPECT_TRUE(pdf2.Init(&buffer.front(), size)); + printing::PdfPsMetafile pdf2(printing::PdfPsMetafile::PDF, + &buffer.front(), + size); // Tries to get the first 4 characters from pdf2. std::vector<char> buffer2(4, 0x00); @@ -59,22 +69,31 @@ TEST(PdfTest, Basic) { TEST(PsTest, Basic) { // Tests in-renderer constructor. printing::PdfPsMetafile ps(printing::PdfPsMetafile::PS); - EXPECT_TRUE(ps.Init()); + cairo_t* context = ps.GetPageContext(); + EXPECT_TRUE(context == NULL); // Renders page 1. - cairo_t* context = ps.StartPage(72, 72); + EXPECT_TRUE(ps.StartPage(72, 72)); + context = ps.GetPageContext(); EXPECT_TRUE(context != NULL); // In theory, we should use Cairo to draw something on |context|. - EXPECT_TRUE(ps.FinishPage(1.5)); + ps.FinishPage(1.5); + context = ps.GetPageContext(); + EXPECT_TRUE(context == NULL); // Renders page 2. - context = ps.StartPage(64, 64); + EXPECT_TRUE(ps.StartPage(64, 64)); + context = ps.GetPageContext(); EXPECT_TRUE(context != NULL); // In theory, we should use Cairo to draw something on |context|. - EXPECT_TRUE(ps.FinishPage(0.5)); + ps.FinishPage(0.5); + context = ps.GetPageContext(); + EXPECT_TRUE(context == NULL); // Closes the file. ps.Close(); + context = ps.GetPageContext(); + EXPECT_TRUE(context == NULL); // Checks data size. unsigned int size = ps.GetDataSize(); @@ -85,8 +104,9 @@ TEST(PsTest, Basic) { ps.GetData(&buffer.front(), size); // Tests another constructor. - printing::PdfPsMetafile ps2(printing::PdfPsMetafile::PS); - EXPECT_TRUE(ps2.Init(&buffer.front(), size)); + printing::PdfPsMetafile ps2(printing::PdfPsMetafile::PS, + &buffer.front(), + size); // Tries to get the first 4 characters from ps2. std::vector<char> buffer2(4, 0x00); |