blob: 7c798ef8337069c27dcc2efc5a1ac311aa545a4a [file] [log] [blame]
reed@google.comdc6c8ba2013-07-18 21:14:04 +00001/*
2 * Copyright 2013 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
reed@google.comeed6f1b2013-07-18 19:53:31 +00008#ifndef SkMipMap_DEFINED
9#define SkMipMap_DEFINED
10
reed9d93c2e2014-10-08 05:17:12 -070011#include "SkCachedData.h"
reed67b09bf2016-01-16 18:50:35 -080012#include "SkPixmap.h"
reed@google.comeed6f1b2013-07-18 19:53:31 +000013#include "SkScalar.h"
fmalita921d7ac2016-01-22 11:45:39 -080014#include "SkSize.h"
reed@google.comeed6f1b2013-07-18 19:53:31 +000015
16class SkBitmap;
reed9d93c2e2014-10-08 05:17:12 -070017class SkDiscardableMemory;
reed@google.comeed6f1b2013-07-18 19:53:31 +000018
reed9d93c2e2014-10-08 05:17:12 -070019typedef SkDiscardableMemory* (*SkDiscardableFactoryProc)(size_t bytes);
20
cblume609623b2016-06-09 09:44:33 -070021/*
22 * SkMipMap will generate mipmap levels when given a base mipmap level image.
23 *
24 * Any function which deals with mipmap levels indices will start with index 0
25 * being the first mipmap level which was generated. Said another way, it does
26 * not include the base level in its range.
27 */
reed9d93c2e2014-10-08 05:17:12 -070028class SkMipMap : public SkCachedData {
reed@google.comeed6f1b2013-07-18 19:53:31 +000029public:
reed67b09bf2016-01-16 18:50:35 -080030 static SkMipMap* Build(const SkPixmap& src, SkDiscardableFactoryProc);
reed9d93c2e2014-10-08 05:17:12 -070031 static SkMipMap* Build(const SkBitmap& src, SkDiscardableFactoryProc);
reed@google.comeed6f1b2013-07-18 19:53:31 +000032
cblumef95ff4a2016-06-02 09:01:48 -070033 // Determines how many levels a SkMipMap will have without creating that mipmap.
cblume609623b2016-06-09 09:44:33 -070034 // This does not include the base mipmap level that the user provided when
35 // creating the SkMipMap.
cblumee2412d52016-02-17 14:53:23 -080036 static int ComputeLevelCount(int baseWidth, int baseHeight);
37
cblumef95ff4a2016-06-02 09:01:48 -070038 // Determines the size of a given mipmap level.
cblume609623b2016-06-09 09:44:33 -070039 // |level| is an index into the generated mipmap levels. It does not include
40 // the base level. So index 0 represents mipmap level 1.
cblume44e09ec2016-06-03 11:59:50 -070041 static SkISize ComputeLevelSize(int baseWidth, int baseHeight, int level);
cblumef95ff4a2016-06-02 09:01:48 -070042
reed@google.comeed6f1b2013-07-18 19:53:31 +000043 struct Level {
reed67b09bf2016-01-16 18:50:35 -080044 SkPixmap fPixmap;
fmalita921d7ac2016-01-22 11:45:39 -080045 SkSize fScale; // < 1.0
reed@google.comeed6f1b2013-07-18 19:53:31 +000046 };
skia.committer@gmail.coma7991982013-07-19 07:00:57 +000047
fmalita33ed3ad2016-02-09 08:20:18 -080048 bool extractLevel(const SkSize& scale, Level*) const;
cblume609623b2016-06-09 09:44:33 -070049
50 // countLevels returns the number of mipmap levels generated (which does not
51 // include the base mipmap level).
cblumee2412d52016-02-17 14:53:23 -080052 int countLevels() const;
cblume609623b2016-06-09 09:44:33 -070053
54 // |index| is an index into the generated mipmap levels. It does not include
55 // the base level. So index 0 represents mipmap level 1.
cblumee2412d52016-02-17 14:53:23 -080056 bool getLevel(int index, Level*) const;
reed@google.comeed6f1b2013-07-18 19:53:31 +000057
reed9d93c2e2014-10-08 05:17:12 -070058protected:
mtklein36352bf2015-03-25 18:17:31 -070059 void onDataChange(void* oldData, void* newData) override {
halcanary96fcdcc2015-08-27 07:41:13 -070060 fLevels = (Level*)newData; // could be nullptr
reed9d93c2e2014-10-08 05:17:12 -070061 }
reed@google.comd94697c2013-07-24 14:31:33 +000062
reed@google.comeed6f1b2013-07-18 19:53:31 +000063private:
64 Level* fLevels;
65 int fCount;
66
67 // we take ownership of levels, and will free it with sk_free()
reed9d93c2e2014-10-08 05:17:12 -070068 SkMipMap(void* malloc, size_t size) : INHERITED(malloc, size) {}
69 SkMipMap(size_t size, SkDiscardableMemory* dm) : INHERITED(size, dm) {}
reed@google.comeed6f1b2013-07-18 19:53:31 +000070
reed9d93c2e2014-10-08 05:17:12 -070071 static size_t AllocLevelsSize(int levelCount, size_t pixelSize);
72
73 typedef SkCachedData INHERITED;
reed@google.comeed6f1b2013-07-18 19:53:31 +000074};
75
76#endif