blob: 0f31a9f703d48904946e142c273918ae508acfbc [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:
reed6644d932016-06-10 11:41:47 -070031 static SkMipMap* Build(const SkPixmap& src, SkSourceGammaTreatment, SkDiscardableFactoryProc);
32 static SkMipMap* Build(const SkBitmap& src, SkSourceGammaTreatment, SkDiscardableFactoryProc);
33
34 static SkSourceGammaTreatment DeduceTreatment(const SkShader::ContextRec& rec) {
35 return (SkShader::ContextRec::kPMColor_DstType == rec.fPreferredDstType) ?
36 SkSourceGammaTreatment::kIgnore : SkSourceGammaTreatment::kRespect;
37 }
reed@google.comeed6f1b2013-07-18 19:53:31 +000038
cblumef95ff4a2016-06-02 09:01:48 -070039 // Determines how many levels a SkMipMap will have without creating that mipmap.
cblume609623b2016-06-09 09:44:33 -070040 // This does not include the base mipmap level that the user provided when
41 // creating the SkMipMap.
cblumee2412d52016-02-17 14:53:23 -080042 static int ComputeLevelCount(int baseWidth, int baseHeight);
43
cblumef95ff4a2016-06-02 09:01:48 -070044 // Determines the size of a given mipmap level.
cblume609623b2016-06-09 09:44:33 -070045 // |level| is an index into the generated mipmap levels. It does not include
46 // the base level. So index 0 represents mipmap level 1.
cblume44e09ec2016-06-03 11:59:50 -070047 static SkISize ComputeLevelSize(int baseWidth, int baseHeight, int level);
cblumef95ff4a2016-06-02 09:01:48 -070048
reed@google.comeed6f1b2013-07-18 19:53:31 +000049 struct Level {
reed67b09bf2016-01-16 18:50:35 -080050 SkPixmap fPixmap;
fmalita921d7ac2016-01-22 11:45:39 -080051 SkSize fScale; // < 1.0
reed@google.comeed6f1b2013-07-18 19:53:31 +000052 };
skia.committer@gmail.coma7991982013-07-19 07:00:57 +000053
fmalita33ed3ad2016-02-09 08:20:18 -080054 bool extractLevel(const SkSize& scale, Level*) const;
cblume609623b2016-06-09 09:44:33 -070055
56 // countLevels returns the number of mipmap levels generated (which does not
57 // include the base mipmap level).
cblumee2412d52016-02-17 14:53:23 -080058 int countLevels() const;
cblume609623b2016-06-09 09:44:33 -070059
60 // |index| is an index into the generated mipmap levels. It does not include
61 // the base level. So index 0 represents mipmap level 1.
cblumee2412d52016-02-17 14:53:23 -080062 bool getLevel(int index, Level*) const;
reed@google.comeed6f1b2013-07-18 19:53:31 +000063
reed9d93c2e2014-10-08 05:17:12 -070064protected:
mtklein36352bf2015-03-25 18:17:31 -070065 void onDataChange(void* oldData, void* newData) override {
halcanary96fcdcc2015-08-27 07:41:13 -070066 fLevels = (Level*)newData; // could be nullptr
reed9d93c2e2014-10-08 05:17:12 -070067 }
reed@google.comd94697c2013-07-24 14:31:33 +000068
reed@google.comeed6f1b2013-07-18 19:53:31 +000069private:
reed6644d932016-06-10 11:41:47 -070070 sk_sp<SkColorSpace> fCS;
71 Level* fLevels; // managed by the baseclass, may be null due to onDataChanged.
72 int fCount;
reed@google.comeed6f1b2013-07-18 19:53:31 +000073
reed9d93c2e2014-10-08 05:17:12 -070074 SkMipMap(void* malloc, size_t size) : INHERITED(malloc, size) {}
75 SkMipMap(size_t size, SkDiscardableMemory* dm) : INHERITED(size, dm) {}
reed@google.comeed6f1b2013-07-18 19:53:31 +000076
reed9d93c2e2014-10-08 05:17:12 -070077 static size_t AllocLevelsSize(int levelCount, size_t pixelSize);
78
79 typedef SkCachedData INHERITED;
reed@google.comeed6f1b2013-07-18 19:53:31 +000080};
81
82#endif