PdfRenderer.cpp revision fce84f035c35606c5707e735f503f7bdcfd5b2a1
1/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "jni.h"
18#include "JNIHelp.h"
19#include "GraphicsJNI.h"
20#include "SkBitmap.h"
21#include "SkMatrix.h"
22#include "fpdfview.h"
23#include "fsdk_rendercontext.h"
24
25#include <android_runtime/AndroidRuntime.h>
26#include <vector>
27#include <utils/Log.h>
28#include <unistd.h>
29#include <sys/types.h>
30#include <unistd.h>
31
32namespace android {
33
34static const int RENDER_MODE_FOR_DISPLAY = 1;
35static const int RENDER_MODE_FOR_PRINT = 2;
36
37static struct {
38    jfieldID x;
39    jfieldID y;
40} gPointClassInfo;
41
42static Mutex sLock;
43
44static int sUnmatchedInitRequestCount = 0;
45
46static void initializeLibraryIfNeeded() {
47    Mutex::Autolock _l(sLock);
48    if (sUnmatchedInitRequestCount == 0) {
49        FPDF_InitLibrary(NULL);
50    }
51    sUnmatchedInitRequestCount++;
52}
53
54static void destroyLibraryIfNeeded() {
55    Mutex::Autolock _l(sLock);
56    sUnmatchedInitRequestCount--;
57    if (sUnmatchedInitRequestCount == 0) {
58       FPDF_DestroyLibrary();
59    }
60}
61
62static int getBlock(void* param, unsigned long position, unsigned char* outBuffer,
63        unsigned long size) {
64    const int fd = reinterpret_cast<intptr_t>(param);
65    const int readCount = pread(fd, outBuffer, size, position);
66    if (readCount < 0) {
67        ALOGE("Cannot read from file descriptor. Error:%d", errno);
68        return 0;
69    }
70    return 1;
71}
72
73static jlong nativeCreate(JNIEnv* env, jclass thiz, jint fd, jlong size) {
74    initializeLibraryIfNeeded();
75
76    FPDF_FILEACCESS loader;
77    loader.m_FileLen = size;
78    loader.m_Param = reinterpret_cast<void*>(intptr_t(fd));
79    loader.m_GetBlock = &getBlock;
80
81    FPDF_DOCUMENT document = FPDF_LoadCustomDocument(&loader, NULL);
82
83    if (!document) {
84        const long error = FPDF_GetLastError();
85        switch (error) {
86            case FPDF_ERR_PASSWORD:
87            case FPDF_ERR_SECURITY: {
88                jniThrowException(env, "java/lang/SecurityException",
89                        "cannot create document. Error:" + error);
90            } break;
91            default: {
92                jniThrowException(env, "java/io/IOException",
93                        "cannot create document. Error:" + error);
94            } break;
95        }
96        destroyLibraryIfNeeded();
97        return -1;
98    }
99
100    return reinterpret_cast<jlong>(document);
101}
102
103static jlong nativeOpenPageAndGetSize(JNIEnv* env, jclass thiz, jlong documentPtr,
104        jint pageIndex, jobject outSize) {
105    FPDF_DOCUMENT document = reinterpret_cast<FPDF_DOCUMENT>(documentPtr);
106
107    FPDF_PAGE page = FPDF_LoadPage(document, pageIndex);
108
109    if (!page) {
110        jniThrowException(env, "java/lang/IllegalStateException",
111                "cannot load page");
112        return -1;
113    }
114
115    double width = 0;
116    double height = 0;
117
118    const int result = FPDF_GetPageSizeByIndex(document, pageIndex, &width, &height);
119
120    if (!result) {
121        jniThrowException(env, "java/lang/IllegalStateException",
122                    "cannot get page size");
123        return -1;
124    }
125
126    env->SetIntField(outSize, gPointClassInfo.x, width);
127    env->SetIntField(outSize, gPointClassInfo.y, height);
128
129    return reinterpret_cast<jlong>(page);
130}
131
132static void nativeClosePage(JNIEnv* env, jclass thiz, jlong pagePtr) {
133    FPDF_PAGE page = reinterpret_cast<FPDF_PAGE>(pagePtr);
134    FPDF_ClosePage(page);
135}
136
137static void nativeClose(JNIEnv* env, jclass thiz, jlong documentPtr) {
138    FPDF_DOCUMENT document = reinterpret_cast<FPDF_DOCUMENT>(documentPtr);
139    FPDF_CloseDocument(document);
140    destroyLibraryIfNeeded();
141}
142
143static jint nativeGetPageCount(JNIEnv* env, jclass thiz, jlong documentPtr) {
144    FPDF_DOCUMENT document = reinterpret_cast<FPDF_DOCUMENT>(documentPtr);
145    return FPDF_GetPageCount(document);
146}
147
148static jboolean nativeScaleForPrinting(JNIEnv* env, jclass thiz, jlong documentPtr) {
149    FPDF_DOCUMENT document = reinterpret_cast<FPDF_DOCUMENT>(documentPtr);
150    return FPDF_VIEWERREF_GetPrintScaling(document);
151}
152
153static void DropContext(void* data) {
154    delete (CRenderContext*) data;
155}
156
157static void renderPageBitmap(FPDF_BITMAP bitmap, FPDF_PAGE page, int destLeft, int destTop,
158        int destRight, int destBottom, SkMatrix* transform, int flags) {
159    // Note: this code ignores the currently unused RENDER_NO_NATIVETEXT,
160    // FPDF_RENDER_LIMITEDIMAGECACHE, FPDF_RENDER_FORCEHALFTONE, FPDF_GRAYSCALE,
161    // and FPDF_ANNOT flags. To add support for that refer to FPDF_RenderPage_Retail
162    // in fpdfview.cpp
163
164    CRenderContext* pContext = FX_NEW CRenderContext;
165
166    CPDF_Page* pPage = (CPDF_Page*) page;
167    pPage->SetPrivateData((void*) 1, pContext, DropContext);
168
169    CFX_FxgeDevice* fxgeDevice = FX_NEW CFX_FxgeDevice;
170    pContext->m_pDevice = fxgeDevice;
171
172    // Reverse the bytes (last argument TRUE) since the Android
173    // format is ARGB while the renderer uses BGRA internally.
174    fxgeDevice->Attach((CFX_DIBitmap*) bitmap, 0, TRUE);
175
176    CPDF_RenderOptions* renderOptions = pContext->m_pOptions;
177
178    if (!renderOptions) {
179        renderOptions = FX_NEW CPDF_RenderOptions;
180        pContext->m_pOptions = renderOptions;
181    }
182
183    if (flags & FPDF_LCD_TEXT) {
184        renderOptions->m_Flags |= RENDER_CLEARTYPE;
185    } else {
186        renderOptions->m_Flags &= ~RENDER_CLEARTYPE;
187    }
188
189    const CPDF_OCContext::UsageType usage = (flags & FPDF_PRINTING)
190            ? CPDF_OCContext::Print : CPDF_OCContext::View;
191
192    renderOptions->m_AddFlags = flags >> 8;
193    renderOptions->m_pOCContext = new CPDF_OCContext(pPage->m_pDocument, usage);
194
195    fxgeDevice->SaveState();
196
197    FX_RECT clip;
198    clip.left = destLeft;
199    clip.right = destRight;
200    clip.top = destTop;
201    clip.bottom = destBottom;
202    fxgeDevice->SetClip_Rect(&clip);
203
204    CPDF_RenderContext* pageContext = FX_NEW CPDF_RenderContext;
205    pContext->m_pContext = pageContext;
206    pageContext->Create(pPage);
207
208    CFX_AffineMatrix matrix;
209    if (!transform) {
210        pPage->GetDisplayMatrix(matrix, destLeft, destTop, destRight - destLeft,
211                destBottom - destTop, 0);
212    } else {
213        // PDF's coordinate system origin is left-bottom while
214        // in graphics it is the top-left, so remap the origin.
215        matrix.Set(1, 0, 0, -1, 0, pPage->GetPageHeight());
216
217        SkScalar transformValues[6];
218        transform->asAffine(transformValues);
219
220        matrix.Concat(transformValues[SkMatrix::kAScaleX], transformValues[SkMatrix::kASkewY],
221                transformValues[SkMatrix::kASkewX], transformValues[SkMatrix::kAScaleY],
222                transformValues[SkMatrix::kATransX], transformValues[SkMatrix::kATransY]);
223    }
224    pageContext->AppendObjectList(pPage, &matrix);
225
226    pContext->m_pRenderer = FX_NEW CPDF_ProgressiveRenderer;
227    pContext->m_pRenderer->Start(pageContext, fxgeDevice, renderOptions, NULL);
228
229    fxgeDevice->RestoreState();
230
231    pPage->RemovePrivateData((void*) 1);
232
233    delete pContext;
234}
235
236static void nativeRenderPage(JNIEnv* env, jclass thiz, jlong documentPtr, jlong pagePtr,
237        jlong bitmapPtr, jint destLeft, jint destTop, jint destRight, jint destBottom,
238        jlong matrixPtr, jint renderMode) {
239
240    FPDF_DOCUMENT document = reinterpret_cast<FPDF_DOCUMENT>(documentPtr);
241    FPDF_PAGE page = reinterpret_cast<FPDF_PAGE>(pagePtr);
242    SkBitmap* skBitmap = reinterpret_cast<SkBitmap*>(bitmapPtr);
243    SkMatrix* skMatrix = reinterpret_cast<SkMatrix*>(matrixPtr);
244
245    skBitmap->lockPixels();
246
247    const int stride = skBitmap->width() * 4;
248
249    FPDF_BITMAP bitmap = FPDFBitmap_CreateEx(skBitmap->width(), skBitmap->height(),
250            FPDFBitmap_BGRA, skBitmap->getPixels(), stride);
251
252    if (!bitmap) {
253        ALOGE("Erorr creating bitmap");
254        return;
255    }
256
257    int renderFlags = 0;
258    if (renderMode == RENDER_MODE_FOR_DISPLAY) {
259        renderFlags |= FPDF_LCD_TEXT;
260    } else if (renderMode == RENDER_MODE_FOR_PRINT) {
261        renderFlags |= FPDF_PRINTING;
262    }
263
264    renderPageBitmap(bitmap, page, destLeft, destTop, destRight,
265            destBottom, skMatrix, renderFlags);
266
267    skBitmap->notifyPixelsChanged();
268    skBitmap->unlockPixels();
269}
270
271static JNINativeMethod gPdfRenderer_Methods[] = {
272    {"nativeCreate", "(IJ)J", (void*) nativeCreate},
273    {"nativeClose", "(J)V", (void*) nativeClose},
274    {"nativeGetPageCount", "(J)I", (void*) nativeGetPageCount},
275    {"nativeScaleForPrinting", "(J)Z", (void*) nativeScaleForPrinting},
276    {"nativeRenderPage", "(JJJIIIIJI)V", (void*) nativeRenderPage},
277    {"nativeOpenPageAndGetSize", "(JILandroid/graphics/Point;)J", (void*) nativeOpenPageAndGetSize},
278    {"nativeClosePage", "(J)V", (void*) nativeClosePage}
279};
280
281int register_android_graphics_pdf_PdfRenderer(JNIEnv* env) {
282    int result = android::AndroidRuntime::registerNativeMethods(
283            env, "android/graphics/pdf/PdfRenderer", gPdfRenderer_Methods,
284            NELEM(gPdfRenderer_Methods));
285
286    jclass clazz = env->FindClass("android/graphics/Point");
287    gPointClassInfo.x = env->GetFieldID(clazz, "x", "I");
288    gPointClassInfo.y = env->GetFieldID(clazz, "y", "I");
289
290    return result;
291};
292
293};
294