xref: /aosp_15_r20/frameworks/base/libs/hwui/jni/FontFamily.cpp (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker  * Copyright (C) 2014 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker  *
4*d57664e9SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker  *
8*d57664e9SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker  *
10*d57664e9SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker  * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker  */
16*d57664e9SAndroid Build Coastguard Worker 
17*d57664e9SAndroid Build Coastguard Worker #include <nativehelper/ScopedPrimitiveArray.h>
18*d57664e9SAndroid Build Coastguard Worker #include <nativehelper/ScopedUtfChars.h>
19*d57664e9SAndroid Build Coastguard Worker #include "FontUtils.h"
20*d57664e9SAndroid Build Coastguard Worker #include "GraphicsJNI.h"
21*d57664e9SAndroid Build Coastguard Worker #include "SkData.h"
22*d57664e9SAndroid Build Coastguard Worker #include "SkFontMgr.h"
23*d57664e9SAndroid Build Coastguard Worker #include "SkRefCnt.h"
24*d57664e9SAndroid Build Coastguard Worker #include "SkStream.h"
25*d57664e9SAndroid Build Coastguard Worker #include "SkTypeface.h"
26*d57664e9SAndroid Build Coastguard Worker #include "Utils.h"
27*d57664e9SAndroid Build Coastguard Worker #include "fonts/Font.h"
28*d57664e9SAndroid Build Coastguard Worker 
29*d57664e9SAndroid Build Coastguard Worker #include <hwui/MinikinSkia.h>
30*d57664e9SAndroid Build Coastguard Worker #include <hwui/Typeface.h>
31*d57664e9SAndroid Build Coastguard Worker #include <minikin/FontFamily.h>
32*d57664e9SAndroid Build Coastguard Worker #include <minikin/LocaleList.h>
33*d57664e9SAndroid Build Coastguard Worker #include <ui/FatVector.h>
34*d57664e9SAndroid Build Coastguard Worker #include <utils/TypefaceUtils.h>
35*d57664e9SAndroid Build Coastguard Worker 
36*d57664e9SAndroid Build Coastguard Worker #include <memory>
37*d57664e9SAndroid Build Coastguard Worker 
38*d57664e9SAndroid Build Coastguard Worker ///////////////////////////////////////////////////////////////////////////////////////////////////
39*d57664e9SAndroid Build Coastguard Worker //
40*d57664e9SAndroid Build Coastguard Worker // The following JNI methods are kept only for compatibility reasons due to hidden API accesses.
41*d57664e9SAndroid Build Coastguard Worker //
42*d57664e9SAndroid Build Coastguard Worker ///////////////////////////////////////////////////////////////////////////////////////////////////
43*d57664e9SAndroid Build Coastguard Worker 
44*d57664e9SAndroid Build Coastguard Worker namespace android {
45*d57664e9SAndroid Build Coastguard Worker 
46*d57664e9SAndroid Build Coastguard Worker namespace {
47*d57664e9SAndroid Build Coastguard Worker struct NativeFamilyBuilder {
NativeFamilyBuilderandroid::__anon0e71180e0111::NativeFamilyBuilder48*d57664e9SAndroid Build Coastguard Worker     NativeFamilyBuilder(uint32_t langId, int variant)
49*d57664e9SAndroid Build Coastguard Worker         : langId(langId), variant(static_cast<minikin::FamilyVariant>(variant)) {}
50*d57664e9SAndroid Build Coastguard Worker     uint32_t langId;
51*d57664e9SAndroid Build Coastguard Worker     minikin::FamilyVariant variant;
52*d57664e9SAndroid Build Coastguard Worker     std::vector<std::shared_ptr<minikin::Font>> fonts;
53*d57664e9SAndroid Build Coastguard Worker     std::vector<minikin::FontVariation> axes;
54*d57664e9SAndroid Build Coastguard Worker };
55*d57664e9SAndroid Build Coastguard Worker }  // namespace
56*d57664e9SAndroid Build Coastguard Worker 
toNativeBuilder(jlong ptr)57*d57664e9SAndroid Build Coastguard Worker static inline NativeFamilyBuilder* toNativeBuilder(jlong ptr) {
58*d57664e9SAndroid Build Coastguard Worker     return reinterpret_cast<NativeFamilyBuilder*>(ptr);
59*d57664e9SAndroid Build Coastguard Worker }
60*d57664e9SAndroid Build Coastguard Worker 
toFamily(jlong ptr)61*d57664e9SAndroid Build Coastguard Worker static inline FontFamilyWrapper* toFamily(jlong ptr) {
62*d57664e9SAndroid Build Coastguard Worker     return reinterpret_cast<FontFamilyWrapper*>(ptr);
63*d57664e9SAndroid Build Coastguard Worker }
64*d57664e9SAndroid Build Coastguard Worker 
toJLong(Ptr ptr)65*d57664e9SAndroid Build Coastguard Worker template<typename Ptr> static inline jlong toJLong(Ptr ptr) {
66*d57664e9SAndroid Build Coastguard Worker     return reinterpret_cast<jlong>(ptr);
67*d57664e9SAndroid Build Coastguard Worker }
68*d57664e9SAndroid Build Coastguard Worker 
FontFamily_initBuilder(JNIEnv * env,jobject clazz,jstring langs,jint variant)69*d57664e9SAndroid Build Coastguard Worker static jlong FontFamily_initBuilder(JNIEnv* env, jobject clazz, jstring langs, jint variant) {
70*d57664e9SAndroid Build Coastguard Worker     NativeFamilyBuilder* builder;
71*d57664e9SAndroid Build Coastguard Worker     if (langs != nullptr) {
72*d57664e9SAndroid Build Coastguard Worker         ScopedUtfChars str(env, langs);
73*d57664e9SAndroid Build Coastguard Worker         builder = new NativeFamilyBuilder(minikin::registerLocaleList(str.c_str()), variant);
74*d57664e9SAndroid Build Coastguard Worker     } else {
75*d57664e9SAndroid Build Coastguard Worker         builder = new NativeFamilyBuilder(minikin::registerLocaleList(""), variant);
76*d57664e9SAndroid Build Coastguard Worker     }
77*d57664e9SAndroid Build Coastguard Worker     return toJLong(builder);
78*d57664e9SAndroid Build Coastguard Worker }
79*d57664e9SAndroid Build Coastguard Worker 
FontFamily_create(CRITICAL_JNI_PARAMS_COMMA jlong builderPtr)80*d57664e9SAndroid Build Coastguard Worker static jlong FontFamily_create(CRITICAL_JNI_PARAMS_COMMA jlong builderPtr) {
81*d57664e9SAndroid Build Coastguard Worker     if (builderPtr == 0) {
82*d57664e9SAndroid Build Coastguard Worker         return 0;
83*d57664e9SAndroid Build Coastguard Worker     }
84*d57664e9SAndroid Build Coastguard Worker     NativeFamilyBuilder* builder = toNativeBuilder(builderPtr);
85*d57664e9SAndroid Build Coastguard Worker     if (builder->fonts.empty()) {
86*d57664e9SAndroid Build Coastguard Worker         return 0;
87*d57664e9SAndroid Build Coastguard Worker     }
88*d57664e9SAndroid Build Coastguard Worker     std::shared_ptr<minikin::FontFamily> family = minikin::FontFamily::create(
89*d57664e9SAndroid Build Coastguard Worker             builder->langId, builder->variant, std::move(builder->fonts),
90*d57664e9SAndroid Build Coastguard Worker             true /* isCustomFallback */, false /* isDefaultFallback */,
91*d57664e9SAndroid Build Coastguard Worker             minikin::VariationFamilyType::None);
92*d57664e9SAndroid Build Coastguard Worker     if (family->getCoverage().length() == 0) {
93*d57664e9SAndroid Build Coastguard Worker         return 0;
94*d57664e9SAndroid Build Coastguard Worker     }
95*d57664e9SAndroid Build Coastguard Worker     return toJLong(new FontFamilyWrapper(std::move(family)));
96*d57664e9SAndroid Build Coastguard Worker }
97*d57664e9SAndroid Build Coastguard Worker 
releaseBuilder(jlong builderPtr)98*d57664e9SAndroid Build Coastguard Worker static void releaseBuilder(jlong builderPtr) {
99*d57664e9SAndroid Build Coastguard Worker     delete toNativeBuilder(builderPtr);
100*d57664e9SAndroid Build Coastguard Worker }
101*d57664e9SAndroid Build Coastguard Worker 
FontFamily_getBuilderReleaseFunc(CRITICAL_JNI_PARAMS)102*d57664e9SAndroid Build Coastguard Worker static jlong FontFamily_getBuilderReleaseFunc(CRITICAL_JNI_PARAMS) {
103*d57664e9SAndroid Build Coastguard Worker     return toJLong(&releaseBuilder);
104*d57664e9SAndroid Build Coastguard Worker }
105*d57664e9SAndroid Build Coastguard Worker 
releaseFamily(jlong familyPtr)106*d57664e9SAndroid Build Coastguard Worker static void releaseFamily(jlong familyPtr) {
107*d57664e9SAndroid Build Coastguard Worker     delete toFamily(familyPtr);
108*d57664e9SAndroid Build Coastguard Worker }
109*d57664e9SAndroid Build Coastguard Worker 
FontFamily_getFamilyReleaseFunc(CRITICAL_JNI_PARAMS)110*d57664e9SAndroid Build Coastguard Worker static jlong FontFamily_getFamilyReleaseFunc(CRITICAL_JNI_PARAMS) {
111*d57664e9SAndroid Build Coastguard Worker     return toJLong(&releaseFamily);
112*d57664e9SAndroid Build Coastguard Worker }
113*d57664e9SAndroid Build Coastguard Worker 
addSkTypeface(NativeFamilyBuilder * builder,sk_sp<SkData> && data,int ttcIndex,jint weight,jint italic)114*d57664e9SAndroid Build Coastguard Worker static bool addSkTypeface(NativeFamilyBuilder* builder, sk_sp<SkData>&& data, int ttcIndex,
115*d57664e9SAndroid Build Coastguard Worker         jint weight, jint italic) {
116*d57664e9SAndroid Build Coastguard Worker     FatVector<SkFontArguments::VariationPosition::Coordinate, 2> skVariation;
117*d57664e9SAndroid Build Coastguard Worker     for (const auto& axis : builder->axes) {
118*d57664e9SAndroid Build Coastguard Worker         skVariation.push_back({axis.axisTag, axis.value});
119*d57664e9SAndroid Build Coastguard Worker     }
120*d57664e9SAndroid Build Coastguard Worker 
121*d57664e9SAndroid Build Coastguard Worker     const size_t fontSize = data->size();
122*d57664e9SAndroid Build Coastguard Worker     const void* fontPtr = data->data();
123*d57664e9SAndroid Build Coastguard Worker     std::unique_ptr<SkStreamAsset> fontData(new SkMemoryStream(std::move(data)));
124*d57664e9SAndroid Build Coastguard Worker 
125*d57664e9SAndroid Build Coastguard Worker     SkFontArguments args;
126*d57664e9SAndroid Build Coastguard Worker     args.setCollectionIndex(ttcIndex);
127*d57664e9SAndroid Build Coastguard Worker     args.setVariationDesignPosition({skVariation.data(), static_cast<int>(skVariation.size())});
128*d57664e9SAndroid Build Coastguard Worker 
129*d57664e9SAndroid Build Coastguard Worker     sk_sp<SkFontMgr> fm = android::FreeTypeFontMgr();
130*d57664e9SAndroid Build Coastguard Worker     sk_sp<SkTypeface> face(fm->makeFromStream(std::move(fontData), args));
131*d57664e9SAndroid Build Coastguard Worker     if (face == NULL) {
132*d57664e9SAndroid Build Coastguard Worker         ALOGE("addFont failed to create font, invalid request");
133*d57664e9SAndroid Build Coastguard Worker         builder->axes.clear();
134*d57664e9SAndroid Build Coastguard Worker         return false;
135*d57664e9SAndroid Build Coastguard Worker     }
136*d57664e9SAndroid Build Coastguard Worker     std::shared_ptr<minikin::MinikinFont> minikinFont = std::make_shared<MinikinFontSkia>(
137*d57664e9SAndroid Build Coastguard Worker             std::move(face), fonts::getNewSourceId(), fontPtr, fontSize, "", ttcIndex,
138*d57664e9SAndroid Build Coastguard Worker             minikin::VariationSettings(builder->axes, false));
139*d57664e9SAndroid Build Coastguard Worker     minikin::Font::Builder fontBuilder(minikinFont);
140*d57664e9SAndroid Build Coastguard Worker 
141*d57664e9SAndroid Build Coastguard Worker     if (weight != RESOLVE_BY_FONT_TABLE) {
142*d57664e9SAndroid Build Coastguard Worker         fontBuilder.setWeight(weight);
143*d57664e9SAndroid Build Coastguard Worker     }
144*d57664e9SAndroid Build Coastguard Worker     if (italic != RESOLVE_BY_FONT_TABLE) {
145*d57664e9SAndroid Build Coastguard Worker         fontBuilder.setSlant(static_cast<minikin::FontStyle::Slant>(italic != 0));
146*d57664e9SAndroid Build Coastguard Worker     }
147*d57664e9SAndroid Build Coastguard Worker     builder->fonts.push_back(fontBuilder.build());
148*d57664e9SAndroid Build Coastguard Worker     builder->axes.clear();
149*d57664e9SAndroid Build Coastguard Worker     return true;
150*d57664e9SAndroid Build Coastguard Worker }
151*d57664e9SAndroid Build Coastguard Worker 
release_global_ref(const void *,void * context)152*d57664e9SAndroid Build Coastguard Worker static void release_global_ref(const void* /*data*/, void* context) {
153*d57664e9SAndroid Build Coastguard Worker     JNIEnv* env = GraphicsJNI::getJNIEnv();
154*d57664e9SAndroid Build Coastguard Worker     bool needToAttach = (env == NULL);
155*d57664e9SAndroid Build Coastguard Worker     if (needToAttach) {
156*d57664e9SAndroid Build Coastguard Worker         env = GraphicsJNI::attachJNIEnv("release_font_data");
157*d57664e9SAndroid Build Coastguard Worker         if (env == nullptr) {
158*d57664e9SAndroid Build Coastguard Worker             ALOGE("failed to attach to thread to release global ref.");
159*d57664e9SAndroid Build Coastguard Worker             return;
160*d57664e9SAndroid Build Coastguard Worker         }
161*d57664e9SAndroid Build Coastguard Worker     }
162*d57664e9SAndroid Build Coastguard Worker 
163*d57664e9SAndroid Build Coastguard Worker     jobject obj = reinterpret_cast<jobject>(context);
164*d57664e9SAndroid Build Coastguard Worker     env->DeleteGlobalRef(obj);
165*d57664e9SAndroid Build Coastguard Worker 
166*d57664e9SAndroid Build Coastguard Worker     if (needToAttach) {
167*d57664e9SAndroid Build Coastguard Worker        GraphicsJNI::detachJNIEnv();
168*d57664e9SAndroid Build Coastguard Worker     }
169*d57664e9SAndroid Build Coastguard Worker }
170*d57664e9SAndroid Build Coastguard Worker 
FontFamily_addFont(JNIEnv * env,jobject clazz,jlong builderPtr,jobject bytebuf,jint ttcIndex,jint weight,jint isItalic)171*d57664e9SAndroid Build Coastguard Worker static jboolean FontFamily_addFont(JNIEnv* env, jobject clazz, jlong builderPtr, jobject bytebuf,
172*d57664e9SAndroid Build Coastguard Worker         jint ttcIndex, jint weight, jint isItalic) {
173*d57664e9SAndroid Build Coastguard Worker     NPE_CHECK_RETURN_ZERO(env, bytebuf);
174*d57664e9SAndroid Build Coastguard Worker     NativeFamilyBuilder* builder = reinterpret_cast<NativeFamilyBuilder*>(builderPtr);
175*d57664e9SAndroid Build Coastguard Worker     const void* fontPtr = env->GetDirectBufferAddress(bytebuf);
176*d57664e9SAndroid Build Coastguard Worker     if (fontPtr == NULL) {
177*d57664e9SAndroid Build Coastguard Worker         ALOGE("addFont failed to create font, buffer invalid");
178*d57664e9SAndroid Build Coastguard Worker         builder->axes.clear();
179*d57664e9SAndroid Build Coastguard Worker         return false;
180*d57664e9SAndroid Build Coastguard Worker     }
181*d57664e9SAndroid Build Coastguard Worker     jlong fontSize = env->GetDirectBufferCapacity(bytebuf);
182*d57664e9SAndroid Build Coastguard Worker     if (fontSize < 0) {
183*d57664e9SAndroid Build Coastguard Worker         ALOGE("addFont failed to create font, buffer size invalid");
184*d57664e9SAndroid Build Coastguard Worker         builder->axes.clear();
185*d57664e9SAndroid Build Coastguard Worker         return false;
186*d57664e9SAndroid Build Coastguard Worker     }
187*d57664e9SAndroid Build Coastguard Worker     jobject fontRef = MakeGlobalRefOrDie(env, bytebuf);
188*d57664e9SAndroid Build Coastguard Worker     sk_sp<SkData> data(SkData::MakeWithProc(fontPtr, fontSize,
189*d57664e9SAndroid Build Coastguard Worker             release_global_ref, reinterpret_cast<void*>(fontRef)));
190*d57664e9SAndroid Build Coastguard Worker     return addSkTypeface(builder, std::move(data), ttcIndex, weight, isItalic);
191*d57664e9SAndroid Build Coastguard Worker }
192*d57664e9SAndroid Build Coastguard Worker 
FontFamily_addFontWeightStyle(JNIEnv * env,jobject clazz,jlong builderPtr,jobject font,jint ttcIndex,jint weight,jint isItalic)193*d57664e9SAndroid Build Coastguard Worker static jboolean FontFamily_addFontWeightStyle(JNIEnv* env, jobject clazz, jlong builderPtr,
194*d57664e9SAndroid Build Coastguard Worker         jobject font, jint ttcIndex, jint weight, jint isItalic) {
195*d57664e9SAndroid Build Coastguard Worker     NPE_CHECK_RETURN_ZERO(env, font);
196*d57664e9SAndroid Build Coastguard Worker     NativeFamilyBuilder* builder = toNativeBuilder(builderPtr);
197*d57664e9SAndroid Build Coastguard Worker     const void* fontPtr = env->GetDirectBufferAddress(font);
198*d57664e9SAndroid Build Coastguard Worker     if (fontPtr == NULL) {
199*d57664e9SAndroid Build Coastguard Worker         ALOGE("addFont failed to create font, buffer invalid");
200*d57664e9SAndroid Build Coastguard Worker         builder->axes.clear();
201*d57664e9SAndroid Build Coastguard Worker         return false;
202*d57664e9SAndroid Build Coastguard Worker     }
203*d57664e9SAndroid Build Coastguard Worker     jlong fontSize = env->GetDirectBufferCapacity(font);
204*d57664e9SAndroid Build Coastguard Worker     if (fontSize < 0) {
205*d57664e9SAndroid Build Coastguard Worker         ALOGE("addFont failed to create font, buffer size invalid");
206*d57664e9SAndroid Build Coastguard Worker         builder->axes.clear();
207*d57664e9SAndroid Build Coastguard Worker         return false;
208*d57664e9SAndroid Build Coastguard Worker     }
209*d57664e9SAndroid Build Coastguard Worker     jobject fontRef = MakeGlobalRefOrDie(env, font);
210*d57664e9SAndroid Build Coastguard Worker     sk_sp<SkData> data(SkData::MakeWithProc(fontPtr, fontSize,
211*d57664e9SAndroid Build Coastguard Worker             release_global_ref, reinterpret_cast<void*>(fontRef)));
212*d57664e9SAndroid Build Coastguard Worker     return addSkTypeface(builder, std::move(data), ttcIndex, weight, isItalic);
213*d57664e9SAndroid Build Coastguard Worker }
214*d57664e9SAndroid Build Coastguard Worker 
FontFamily_addAxisValue(CRITICAL_JNI_PARAMS_COMMA jlong builderPtr,jint tag,jfloat value)215*d57664e9SAndroid Build Coastguard Worker static void FontFamily_addAxisValue(CRITICAL_JNI_PARAMS_COMMA jlong builderPtr, jint tag, jfloat value) {
216*d57664e9SAndroid Build Coastguard Worker     NativeFamilyBuilder* builder = toNativeBuilder(builderPtr);
217*d57664e9SAndroid Build Coastguard Worker     builder->axes.push_back({static_cast<minikin::AxisTag>(tag), value});
218*d57664e9SAndroid Build Coastguard Worker }
219*d57664e9SAndroid Build Coastguard Worker 
220*d57664e9SAndroid Build Coastguard Worker ///////////////////////////////////////////////////////////////////////////////
221*d57664e9SAndroid Build Coastguard Worker 
222*d57664e9SAndroid Build Coastguard Worker static const JNINativeMethod gFontFamilyMethods[] = {
223*d57664e9SAndroid Build Coastguard Worker     { "nInitBuilder",           "(Ljava/lang/String;I)J", (void*)FontFamily_initBuilder },
224*d57664e9SAndroid Build Coastguard Worker     { "nCreateFamily",          "(J)J", (void*)FontFamily_create },
225*d57664e9SAndroid Build Coastguard Worker     { "nGetBuilderReleaseFunc", "()J", (void*)FontFamily_getBuilderReleaseFunc },
226*d57664e9SAndroid Build Coastguard Worker     { "nGetFamilyReleaseFunc",  "()J", (void*)FontFamily_getFamilyReleaseFunc },
227*d57664e9SAndroid Build Coastguard Worker     { "nAddFont",               "(JLjava/nio/ByteBuffer;III)Z", (void*)FontFamily_addFont },
228*d57664e9SAndroid Build Coastguard Worker     { "nAddFontWeightStyle",    "(JLjava/nio/ByteBuffer;III)Z",
229*d57664e9SAndroid Build Coastguard Worker             (void*)FontFamily_addFontWeightStyle },
230*d57664e9SAndroid Build Coastguard Worker     { "nAddAxisValue",         "(JIF)V", (void*)FontFamily_addAxisValue },
231*d57664e9SAndroid Build Coastguard Worker };
232*d57664e9SAndroid Build Coastguard Worker 
register_android_graphics_FontFamily(JNIEnv * env)233*d57664e9SAndroid Build Coastguard Worker int register_android_graphics_FontFamily(JNIEnv* env)
234*d57664e9SAndroid Build Coastguard Worker {
235*d57664e9SAndroid Build Coastguard Worker     int err = RegisterMethodsOrDie(env, "android/graphics/FontFamily", gFontFamilyMethods,
236*d57664e9SAndroid Build Coastguard Worker             NELEM(gFontFamilyMethods));
237*d57664e9SAndroid Build Coastguard Worker 
238*d57664e9SAndroid Build Coastguard Worker     init_FontUtils(env);
239*d57664e9SAndroid Build Coastguard Worker     return err;
240*d57664e9SAndroid Build Coastguard Worker }
241*d57664e9SAndroid Build Coastguard Worker 
242*d57664e9SAndroid Build Coastguard Worker }
243