亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

解析android截屏問題

2020-04-11 12:04:07
字體:
來源:轉載
供稿:網友
我是基于android2.3.3系統之上的,想必大家應該知道在android源碼下面有個文件叫做screencap吧,位于frameworks/base/services/surfaceflinger/tests/screencap/screencap.cpp,你直接在linux下編譯(保存在 /system/bin/test-screencap),然后push到手機上再通過電腦去敲命令test-screencap /mnt/sdcard/scapxx.png就可以實現截屏。
復制代碼 代碼如下:

/*
  * Copyright (C) 2010 The Android Open Source Project
  *
  * Licensed under the Apache License, Version 2.0 (the "License");
  * you may not use this file except in compliance with the License.
  * You may obtain a copy of the License at
  *
  *      http://www.apache.org/licenses/LICENSE-2.0
  *
  * Unless required by applicable law or agreed to in writing, software
  * distributed under the License is distributed on an "AS IS" BASIS,
  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  * See the License for the specific language governing permissions and
  * limitations under the License.
  */

 #include <utils/Log.h>

 #include <binder/IPCThreadState.h>
 #include <binder/ProcessState.h>
 #include <binder/IServiceManager.h>

 #include <binder/IMemory.h>
 #include <surfaceflinger/ISurfaceComposer.h>

 #include <SkImageEncoder.h>
 #include <SkBitmap.h>

 using namespace android;

 int main(int argc, char** argv)
 {
     if (argc != 2) {
         printf("usage: %s path/n", argv[0]);
         exit(0);
     }

     const String16 name("SurfaceFlinger");
     sp<ISurfaceComposer> composer;
     getService(name, &composer);

     sp<IMemoryHeap> heap;
     uint32_t w, h;
     PixelFormat f;
     status_t err = composer->captureScreen(0, &heap, &w, &h, &f, 0, 0);
     if (err != NO_ERROR) {
         fprintf(stderr, "screen capture failed: %s/n", strerror(-err));
         exit(0);
     }

     printf("screen capture success: w=%u, h=%u, pixels=%p/n",
             w, h, heap->getBase());

     printf("saving file as PNG in %s .../n", argv[1]);

     SkBitmap b;
     b.setConfig(SkBitmap::kARGB_8888_Config, w, h);
     b.setPixels(heap->getBase());
     SkImageEncoder::EncodeFile(argv[1], b,
             SkImageEncoder::kPNG_Type, SkImageEncoder::kDefaultQuality);

     return 0;
 }

其實這個程序真正用到的就是一個叫做capturescreen的函數,而capturescreen會調用captureScreenImplLocked這個函數
下面是代碼:
復制代碼 代碼如下:

status_t SurfaceFlinger::captureScreenImplLocked(DisplayID dpy,
         sp<IMemoryHeap>* heap,
         uint32_t* w, uint32_t* h, PixelFormat* f,
         uint32_t sw, uint32_t sh)
 {
    LOGI("captureScreenImplLocked");
     status_t result = PERMISSION_DENIED;

     // only one display supported for now
     if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
         return BAD_VALUE;

     if (!GLExtensions::getInstance().haveFramebufferObject())
         return INVALID_OPERATION;

     // get screen geometry
     const DisplayHardware& hw(graphicPlane(dpy).displayHardware());
     const uint32_t hw_w = hw.getWidth();
     const uint32_t hw_h = hw.getHeight();

     if ((sw > hw_w) || (sh > hw_h))
         return BAD_VALUE;

     sw = (!sw) ? hw_w : sw;
     sh = (!sh) ? hw_h : sh;
     const size_t size = sw * sh * 4;

     // make sure to clear all GL error flags
     while ( glGetError() != GL_NO_ERROR ) ;

     // create a FBO
     GLuint name, tname;
     glGenRenderbuffersOES(1, &tname);
     glBindRenderbufferOES(GL_RENDERBUFFER_OES, tname);
     glRenderbufferStorageOES(GL_RENDERBUFFER_OES, GL_RGBA8_OES, sw, sh);
     glGenFramebuffersOES(1, &name);
     glBindFramebufferOES(GL_FRAMEBUFFER_OES, name);
     glFramebufferRenderbufferOES(GL_FRAMEBUFFER_OES,
             GL_COLOR_ATTACHMENT0_OES, GL_RENDERBUFFER_OES, tname);

     GLenum status = glCheckFramebufferStatusOES(GL_FRAMEBUFFER_OES);
     if (status == GL_FRAMEBUFFER_COMPLETE_OES) {

         // invert everything, b/c glReadPixel() below will invert the FB
         glViewport(0, 0, sw, sh);
         glScissor(0, 0, sw, sh);
         glMatrixMode(GL_PROJECTION);
         glPushMatrix();
         glLoadIdentity();
         glOrthof(0, hw_w, 0, hw_h, 0, 1);
         glMatrixMode(GL_MODELVIEW);

         // redraw the screen entirely...
         glClearColor(0,0,0,1);
         glClear(GL_COLOR_BUFFER_BIT);

         const Vector< sp<LayerBase> >& layers(mVisibleLayersSortedByZ);
         const size_t count = layers.size();
         for (size_t i=0 ; i<count ; ++i) {
             const sp<LayerBase>& layer(layers[i]);
             layer->drawForSreenShot();
         }

         // XXX: this is needed on tegra
         glScissor(0, 0, sw, sh);

         // check for errors and return screen capture
         if (glGetError() != GL_NO_ERROR) {
             // error while rendering
             result = INVALID_OPERATION;
         } else {
             // allocate shared memory large enough to hold the
             // screen capture
             sp<MemoryHeapBase> base(
                     new MemoryHeapBase(size, 0, "screen-capture") );
             void* const ptr = base->getBase();
             if (ptr) {
                 // capture the screen with glReadPixels()
                 glReadPixels(0, 0, sw, sh, GL_RGBA, GL_UNSIGNED_BYTE, ptr);
                 if (glGetError() == GL_NO_ERROR) {
                     *heap = base;
                     *w = sw;
                     *h = sh;
                     *f = PIXEL_FORMAT_RGBA_8888;
                     result = NO_ERROR;
                 }
             } else {
                 result = NO_MEMORY;
             }
         }
         glEnable(GL_SCISSOR_TEST);
         glViewport(0, 0, hw_w, hw_h);
         glMatrixMode(GL_PROJECTION);
         glPopMatrix();
         glMatrixMode(GL_MODELVIEW);

  
     } else {
         result = BAD_VALUE;
     }

     // release FBO resources
     glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0);
     glDeleteRenderbuffersOES(1, &tname);
     glDeleteFramebuffersOES(1, &name);

     hw.compositionComplete();

     return result;
 }

  
 status_t SurfaceFlinger::captureScreen(DisplayID dpy,
         sp<IMemoryHeap>* heap,
         uint32_t* width, uint32_t* height, PixelFormat* format,
         uint32_t sw, uint32_t sh)
 {
            LOGI("into captureScreen");          
     // only one display supported for now
     if (UNLIKELY(uint32_t(dpy) >= DISPLAY_COUNT))
         return BAD_VALUE;

     if (!GLExtensions::getInstance().haveFramebufferObject())
         return INVALID_OPERATION;

     class MessageCaptureScreen : public MessageBase {
         SurfaceFlinger* flinger;
         DisplayID dpy;
         sp<IMemoryHeap>* heap;
         uint32_t* w;
         uint32_t* h;
         PixelFormat* f;
         uint32_t sw;
         uint32_t sh;
         status_t result;
     public:
         MessageCaptureScreen(SurfaceFlinger* flinger, DisplayID dpy,
                 sp<IMemoryHeap>* heap, uint32_t* w, uint32_t* h, PixelFormat* f,
                 uint32_t sw, uint32_t sh)
             : flinger(flinger), dpy(dpy),
               heap(heap), w(w), h(h), f(f), sw(sw), sh(sh), result(PERMISSION_DENIED)
         {

         }
         status_t getResult() const {
            LOGI("getResult()");          
          return result;
         }
         virtual bool handler() {

  
    LOGI("handler()");
             Mutex::Autolock _l(flinger->mStateLock);

             // if we have secure windows, never allow the screen capture
             if (flinger->mSecureFrameBuffer)
                 return true;

             result = flinger->captureScreenImplLocked(dpy,
                     heap, w, h, f, sw, sh);

             return true;
         }
     };
    LOGI("before messagecapturescreen");
     sp<MessageBase> msg = new MessageCaptureScreen(this,
             dpy, heap, width, height, format, sw, sh);
     status_t res = postMessageSync(msg);
     if (res == NO_ERROR) {
         res = static_cast<MessageCaptureScreen*>( msg.get() )->getResult();
     }
     return res;
 }

而這個函數關鍵又使用了opengl的幾個函數去獲得圖片,然而opengl又去read framebuffer(這是我的理解)。如果你去用jni調用so的方法去截屏的話,就可以把screencap這個文件稍微修改一下然后做成so文件。

主要是補充一下怎么去存放文件與編譯吧,當然我說的方法只是我做的方法不代表是很好用的。

存放:在eclipse新建一個android工程,保存后找到這個工程(如screencap)的存放位置 然后把這個文件放到android源代碼的development文件里面,然后在你的那個工程文件里面新建一個文件夾,名字叫做jni(這個文件夾平行于src文件夾,screencap/jni),把上面博客提到的那個C++跟mk(screencap/jni/com_android_ScreenCap_ScreenCapNative.cpp和screencap/jni/Android.mk)文件放進去,最后在把編譯的mk文件放在screencap目錄下(screencap/Android.mk);

編譯:編譯是個很偉大的工程,需要你花大量的時間與精力。直接在終端進入工程存放的所在位置,我的是Administrator/Android.2.3.3/development,然后mm(Builds all of the modules in the current directory),如果成功,那么你運氣比較好,在終端回提示你APK保存的位置。push進手機試一試。但是往往是不成功的。你可能會遇到一些問題,比如android.permission.ACCESS_SURFACE_FLINGER ,android.permission.READ_FRAME_BUFFER(因為capturescrren這個函數是surfaceflinger里面的函數,然而surfaceflinger里面的opengl截屏函數會去讀取framebuffer),相關源代碼是:
復制代碼 代碼如下:

status_t SurfaceFlinger::onTransact(
     uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
 {

     switch (code) {
         case CREATE_CONNECTION:
         case OPEN_GLOBAL_TRANSACTION:
         case CLOSE_GLOBAL_TRANSACTION:
         case SET_ORIENTATION:
         case FREEZE_DISPLAY:
         case UNFREEZE_DISPLAY:
         case BOOT_FINISHED:
         case TURN_ELECTRON_BEAM_OFF:
         case TURN_ELECTRON_BEAM_ON:
         {
             // codes that require permission check
             IPCThreadState* ipc = IPCThreadState::self();
             const int pid = ipc->getCallingPid();
             const int uid = ipc->getCallingUid();
             if ((uid != AID_GRAPHICS) && !mAccessSurfaceFlinger.check(pid, uid)) {
                 LOGE("Permission Denial: "
                         "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
                 return PERMISSION_DENIED;
             }
             break;
         }
         case CAPTURE_SCREEN:
         {
             // codes that require permission check
             IPCThreadState* ipc = IPCThreadState::self();
             const int pid = ipc->getCallingPid();
             const int uid = ipc->getCallingUid();
             if ((uid != AID_GRAPHICS) && !mReadFramebuffer.check(pid, uid)) {

                 LOGE("Permission Denial: "
                         "can't read framebuffer pid=%d, uid=%d", pid, uid);
                 return PERMISSION_DENIED;
             }

             break;
         }
     }

     status_t err = BnSurfaceComposer::onTransact(code, data, reply, flags);
     if (err == UNKNOWN_TRANSACTION || err == PERMISSION_DENIED) {
         CHECK_INTERFACE(ISurfaceComposer, data, reply);
         if (UNLIKELY(!mHardwareTest.checkCalling())) {
             IPCThreadState* ipc = IPCThreadState::self();
             const int pid = ipc->getCallingPid();
             const int uid = ipc->getCallingUid();

    LOGI("err");
             LOGE("Permission Denial: "
                     "can't access SurfaceFlinger pid=%d, uid=%d", pid, uid);
             return PERMISSION_DENIED;
         }
         int n;
         switch (code) {
             case 1000: // SHOW_CPU, NOT SUPPORTED ANYMORE
             case 1001: // SHOW_FPS, NOT SUPPORTED ANYMORE
                 return NO_ERROR;
             case 1002:  // SHOW_UPDATES
                 n = data.readInt32();
                 mDebugRegion = n ? n : (mDebugRegion ? 0 : 1);
                 return NO_ERROR;
             case 1003:  // SHOW_BACKGROUND
                 n = data.readInt32();
                 mDebugBackground = n ? 1 : 0;
                 return NO_ERROR;
             case 1004:{ // repaint everything
                 Mutex::Autolock _l(mStateLock);
                 const DisplayHardware& hw(graphicPlane(0).displayHardware());
                 mDirtyRegion.set(hw.bounds()); // careful that's not thread-safe
                 signalEvent();
                 return NO_ERROR;
             }
             case 1005:{ // force transaction
                 setTransactionFlags(eTransactionNeeded|eTraversalNeeded);
                 return NO_ERROR;
             }
             case 1006:{ // enable/disable GraphicLog
                 int enabled = data.readInt32();
                 GraphicLog::getInstance().setEnabled(enabled);
                 return NO_ERROR;
             }
             case 1007: // set mFreezeCount
                 mFreezeCount = data.readInt32();
                 mFreezeDisplayTime = 0;
                 return NO_ERROR;
             case 1010:  // interrogate.
                 reply->writeInt32(0);
                 reply->writeInt32(0);
                 reply->writeInt32(mDebugRegion);
                 reply->writeInt32(mDebugBackground);
                 return NO_ERROR;
             case 1013: {
                 Mutex::Autolock _l(mStateLock);
                 const DisplayHardware& hw(graphicPlane(0).displayHardware());
                 reply->writeInt32(hw.getPageFlipCount());
             }
             return NO_ERROR;
         }
     }
     return err;
 }

這個僅僅只是開始!  你會發現你即使在xml里面添加相應的權限仍然會有這個問題出現,為什么呢?在packageManger文件里面發現相關代碼:
復制代碼 代碼如下:

 int checkSignaturesLP(Signature[] s1, Signature[] s2) {
        if (s1 == null) {
            return s2 == null
                    ? PackageManager.SIGNATURE_NEITHER_SIGNED
                    : PackageManager.SIGNATURE_FIRST_NOT_SIGNED;
        }
        if (s2 == null) {
            return PackageManager.SIGNATURE_SECOND_NOT_SIGNED;
        }
        HashSet<Signature> set1 = new HashSet<Signature>();
        for (Signature sig : s1) {
            set1.add(sig);
        }
        HashSet<Signature> set2 = new HashSet<Signature>();
        for (Signature sig : s2) {
            set2.add(sig);
        }
        // Make sure s2 contains all signatures in s1.
        if (set1.equals(set2)) {
            return PackageManager.SIGNATURE_MATCH;
        }
        return PackageManager.SIGNATURE_NO_MATCH;
    }

  

 // Check for shared user signatures
        if (pkgSetting.sharedUser != null && pkgSetting.sharedUser.signatures.mSignatures != null) {
            if (checkSignaturesLP(pkgSetting.sharedUser.signatures.mSignatures,
                    pkg.mSignatures) != PackageManager.SIGNATURE_MATCH) {
                Slog.e(TAG, "Package " + pkg.packageName
                        + " has no signatures that match those in shared user "
                        + pkgSetting.sharedUser.name + "; ignoring!");
                mLastScanError = PackageManager.INSTALL_FAILED_SHARED_USER_INCOMPATIBLE;
                return false;
            }
        }
        return true;

  
    private boolean verifySignaturesLP(PackageSetting pkgSetting,
            PackageParser.Package pkg) {

        // Check for shared user signatures
        if (pkgSetting.sharedUser != null && pkgSetting.sharedUser.signatures.mSignatures != null) {

        if (checkSignaturesLP(pkgSetting.sharedUser.signatures.mSignatures,
                    pkg.mSignatures) != PackageManager.SIGNATURE_MATCH) {

            Slog.e(TAG, "Package " + pkg.packageName
                        + " has no signatures that match those in shared user "
                        + pkgSetting.sharedUser.name + "; ignoring!");
                mLastScanError = PackageManager.INSTALL_FAILED_SHARED_USER_INCOMPATIBLE;
                return false;

            }
        }
        return true;
    }

你在終端輸入adb logcat | grep PackageManager 你會發現這兩個權限根本沒有賦予給你的apk,我的理解是,程序需要權限,然后apk仍然需要權限。那怎么樣給apk賦予權限呢,兩個方法,一個是在我上面說的screencap/Android.mk里面添加platform一行,然后在回到mm。還有一個方法就是通過sign。這兩個方法都是給apk賦予system權限,但是我試過這兩種方法,都有問題,就是在adb install的時候會顯示簽名不兼容,查看源代碼會發現uid跟gid不匹配。這些是我這段時間發現的問題,大家有問題可以交流交流。
再說說幾個簡單的應用層截屏吧,很簡單,就是幾個函數調用而已
復制代碼 代碼如下:

View view = getWindow().getDecorView();  
       Display display = this.getWindowManager().getDefaultDisplay();  
       view.layout(0, 0, display.getWidth(), display.getHeight());  
       view.setDrawingCacheEnabled(true);//允許當前窗口保存緩存信息,這樣  getDrawingCache()方法才會返回一個Bitmap  
       Bitmap bmp = Bitmap.createBitmap(view.getDrawingCache());

我對這個程序的理解就是,它僅僅只能截取當前的activity,也就是說如果你運行這個程序后它就截取你這個程序的當前屏幕的信息。我們假設你做成一個按鈕,在點擊按鈕后sleep5秒再調用這個方法(假設你的activity叫做screen)。當你點擊按鈕以后,然后你再點擊home或者返回按鈕,等到5秒后你那個程序就會截取到你當前屏幕?不是!它只會截取那個運行于后臺的screen這個activity。

這些只是我的一點小小的總結,而且肯定有不對的地方,希望大家一起來解決截屏的問題!
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美高清视频一区二区| 91国偷自产一区二区三区的观看方式| 国产视频自拍一区| 日韩男女性生活视频| 久久天堂av综合合色| 伊人青青综合网站| 久久久久五月天| 国外成人在线播放| 午夜精品久久久久久久99黑人| 爽爽爽爽爽爽爽成人免费观看| 中文字幕亚洲在线| 伊人一区二区三区久久精品| 国产精品自拍偷拍| 日韩电影免费观看中文字幕| 日韩在线观看免费高清完整版| 欧美性高跟鞋xxxxhd| 国产精品视频一区国模私拍| 久久精品亚洲国产| 欧美大片在线看| 在线精品国产欧美| 色在人av网站天堂精品| 日韩免费在线播放| 91av在线免费观看视频| 2018日韩中文字幕| 国产精品久久久久久久app| 欧美成人午夜剧场免费观看| 色午夜这里只有精品| 91精品免费视频| 欧美激情精品久久久久久免费印度| 欧美亚洲一区在线| 亚洲国产精品嫩草影院久久| 欧美野外wwwxxx| 亚洲人精选亚洲人成在线| 中文字幕精品av| 欧美激情综合亚洲一二区| 成人精品一区二区三区电影免费| 不卡毛片在线看| 91九色视频导航| 亚洲人成网站免费播放| 热久久视久久精品18亚洲精品| 久久精品国产久精国产思思| 亚洲欧洲在线免费| 理论片在线不卡免费观看| 日韩人体视频一二区| 国产精品久久久久国产a级| 一区二区三区回区在观看免费视频| 91在线视频一区| 国产精品jvid在线观看蜜臀| 久久精品在线视频| 欧美视频裸体精品| 久久影视电视剧免费网站清宫辞电视| 欧美丝袜一区二区三区| 91产国在线观看动作片喷水| 亚洲精品久久7777777| 国产福利精品视频| 久久久免费观看视频| 久久人人爽人人爽人人片av高清| 中文国产成人精品久久一| 欧美成人激情图片网| 欧美亚洲在线观看| 国产精品老女人视频| 久久影视三级福利片| 国产剧情久久久久久| 日韩中文字幕视频| 一道本无吗dⅴd在线播放一区| 国产午夜精品理论片a级探花| 国产精品999999| 成人激情在线观看| 在线亚洲男人天堂| 久久露脸国产精品| 久久av在线播放| 成人春色激情网| 国产日韩精品视频| 亚洲精品久久久久久下一站| 亚洲成年网站在线观看| 亚洲女人天堂色在线7777| 另类专区欧美制服同性| 97精品久久久| 国产精品久久中文| 国产一区二区三区高清在线观看| www.日本久久久久com.| 亚洲国产日韩精品在线| 欧美中文字幕视频在线观看| 日韩美女av在线| 国模gogo一区二区大胆私拍| 久久国产精品久久久久| 国产亚洲人成网站在线观看| 91免费精品国偷自产在线| 精品调教chinesegay| 色综合亚洲精品激情狠狠| 国产经典一区二区| 亚洲国产天堂久久国产91| 欧美在线观看日本一区| 亚洲一区二区免费在线| 日韩一区二区福利| 91av在线播放| 日本欧美爱爱爱| 国产精品高潮呻吟久久av野狼| 亚洲综合在线中文字幕| 亚洲男人天堂古典| 欧美成aaa人片在线观看蜜臀| 亚洲欧美日韩国产中文专区| 欧美高清一级大片| 国产成人精彩在线视频九色| 国产亚洲精品日韩| 日韩成人在线视频观看| 亚洲国产精品va在线看黑人| 欧美大片在线免费观看| 欧美精品激情在线| 中日韩美女免费视频网站在线观看| 亚洲最大成人网色| 国产欧美在线观看| 欧美国产第一页| 亚洲国产日韩精品在线| 精品久久久av| 欧美www视频在线观看| 欧美性做爰毛片| 日韩精品免费观看| 国产精品永久免费在线| 久久五月情影视| 大桥未久av一区二区三区| 性色av香蕉一区二区| 国产欧美一区二区三区久久| 国产亚洲精品综合一区91| 欧美精品生活片| 日韩av中文在线| 久久av在线播放| 自拍视频国产精品| 亚洲aⅴ日韩av电影在线观看| 国产精品中文在线| 亚洲欧美日韩在线一区| 久久久999精品免费| 亚洲精品自拍偷拍| 美日韩丰满少妇在线观看| 国产精品最新在线观看| 亚洲一级黄色片| 日韩激情片免费| 亚洲免费一级电影| 欧美成人网在线| 日韩成人av网址| 91夜夜揉人人捏人人添红杏| 国产精品电影在线观看| 激情懂色av一区av二区av| 欧美日本中文字幕| 亚洲在线免费看| 久久久综合免费视频| 2019中文字幕全在线观看| 欧美国产视频一区二区| 欧美有码在线观看| 成人精品久久一区二区三区| 亚洲欧美999| 日韩视频精品在线| 欧美性猛交xxxx乱大交| 欧美视频第一页| 久久视频国产精品免费视频在线| 欧美一区二区大胆人体摄影专业网站| 亚洲男人天堂九九视频| 午夜精品福利电影| 日韩精品中文字幕有码专区| 亚洲在线免费看| 欧美电影在线观看高清| 亚洲天堂av在线免费观看| 久久69精品久久久久久久电影好| 少妇高潮久久久久久潘金莲|