blob: f3425cbeaa56763052fb44c1c949aa485fcb61e9 [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"
reed6644d932016-06-10 11:41:47 -070015#include "SkShader.h"
reed@google.comeed6f1b2013-07-18 19:53:31 +000016
17class SkBitmap;
reed9d93c2e2014-10-08 05:17:12 -070018class SkDiscardableMemory;
reed@google.comeed6f1b2013-07-18 19:53:31 +000019
reed9d93c2e2014-10-08 05:17:12 -070020typedef SkDiscardableMemory* (*SkDiscardableFactoryProc)(size_t bytes);
21
cblume609623b2016-06-09 09:44:33 -070022/*
23 * SkMipMap will generate mipmap levels when given a base mipmap level image.
24 *
25 * Any function which deals with mipmap levels indices will start with index 0
26 * being the first mipmap level which was generated. Said another way, it does
27 * not include the base level in its range.
28 */
reed9d93c2e2014-10-08 05:17:12 -070029class SkMipMap : public SkCachedData {
reed@google.comeed6f1b2013-07-18 19:53:31 +000030public:
Brian Osman7b8400d2016-11-08 17:08:54 -050031 static SkMipMap* Build(const SkPixmap& src, SkDestinationSurfaceColorMode,
32 SkDiscardableFactoryProc);
33 static SkMipMap* Build(const SkBitmap& src, SkDestinationSurfaceColorMode,
34 SkDiscardableFactoryProc);
reed6644d932016-06-10 11:41:47 -070035
Brian Osman7b8400d2016-11-08 17:08:54 -050036 static SkDestinationSurfaceColorMode DeduceColorMode(const SkShader::ContextRec& rec) {
37 return (SkShader::ContextRec::kPMColor_DstType == rec.fPreferredDstType)
38 ? SkDestinationSurfaceColorMode::kLegacy
39 : SkDestinationSurfaceColorMode::kGammaAndColorSpaceAware;
reed6644d932016-06-10 11:41:47 -070040 }
reed@google.comeed6f1b2013-07-18 19:53:31 +000041
cblumef95ff4a2016-06-02 09:01:48 -070042 // Determines how many levels a SkMipMap will have without creating that mipmap.
cblume609623b2016-06-09 09:44:33 -070043 // This does not include the base mipmap level that the user provided when
44 // creating the SkMipMap.
cblumee2412d52016-02-17 14:53:23 -080045 static int ComputeLevelCount(int baseWidth, int baseHeight);
46
cblumef95ff4a2016-06-02 09:01:48 -070047 // Determines the size of a given mipmap level.
cblume609623b2016-06-09 09:44:33 -070048 // |level| is an index into the generated mipmap levels. It does not include
49 // the base level. So index 0 represents mipmap level 1.
cblume44e09ec2016-06-03 11:59:50 -070050 static SkISize ComputeLevelSize(int baseWidth, int baseHeight, int level);
cblumef95ff4a2016-06-02 09:01:48 -070051
reed@google.comeed6f1b2013-07-18 19:53:31 +000052 struct Level {
reed67b09bf2016-01-16 18:50:35 -080053 SkPixmap fPixmap;
fmalita921d7ac2016-01-22 11:45:39 -080054 SkSize fScale; // < 1.0
reed@google.comeed6f1b2013-07-18 19:53:31 +000055 };
skia.committer@gmail.coma7991982013-07-19 07:00:57 +000056
fmalita33ed3ad2016-02-09 08:20:18 -080057 bool extractLevel(const SkSize& scale, Level*) const;
cblume609623b2016-06-09 09:44:33 -070058
59 // countLevels returns the number of mipmap levels generated (which does not
60 // include the base mipmap level).
cblumee2412d52016-02-17 14:53:23 -080061 int countLevels() const;
cblume609623b2016-06-09 09:44:33 -070062
63 // |index| is an index into the generated mipmap levels. It does not include
64 // the base level. So index 0 represents mipmap level 1.
cblumee2412d52016-02-17 14:53:23 -080065 bool getLevel(int index, Level*) const;
reed@google.comeed6f1b2013-07-18 19:53:31 +000066
reed9d93c2e2014-10-08 05:17:12 -070067protected:
mtklein36352bf2015-03-25 18:17:31 -070068 void onDataChange(void* oldData, void* newData) override {
halcanary96fcdcc2015-08-27 07:41:13 -070069 fLevels = (Level*)newData; // could be nullptr
reed9d93c2e2014-10-08 05:17:12 -070070 }
reed@google.comd94697c2013-07-24 14:31:33 +000071
reed@google.comeed6f1b2013-07-18 19:53:31 +000072private:
reed6644d932016-06-10 11:41:47 -070073 sk_sp<SkColorSpace> fCS;
74 Level* fLevels; // managed by the baseclass, may be null due to onDataChanged.
75 int fCount;
reed@google.comeed6f1b2013-07-18 19:53:31 +000076
reed9d93c2e2014-10-08 05:17:12 -070077 SkMipMap(void* malloc, size_t size) : INHERITED(malloc, size) {}
78 SkMipMap(size_t size, SkDiscardableMemory* dm) : INHERITED(size, dm) {}
reed@google.comeed6f1b2013-07-18 19:53:31 +000079
reed9d93c2e2014-10-08 05:17:12 -070080 static size_t AllocLevelsSize(int levelCount, size_t pixelSize);
81
82 typedef SkCachedData INHERITED;
reed@google.comeed6f1b2013-07-18 19:53:31 +000083};
84
85#endif