sylware / charfbuzz (public) (License: LGPLv3) (since 2019-09-09) (hash sha1)
partial C implementation of harfbuzz C API for roman scripts

/hb-shape-plan.c (11f405cdb548fd9ac6aaebda60059202512d5831) (14266 bytes) (mode 100644) (type blob)

/*
Port from c++ is protected by a GNU Lesser GPLv3
Copyright © 2013 Sylvain BERTRAND <sylvain.bertrand@gmail.com>
                                  <sylware@legeek.net>       
*/
#include <stddef.h>
#include <assert.h>

#include <ft2build.h>
#include FT_FREETYPE_H

#include "hb.h"
#include "hb-private.h"
#include "hb-atomic-private.h"
#include "hb-buffer-private.h"
#include "hb-shaper-private.h"
#include "hb-face-private.h"
#include "hb-font-private.h"
#include "hb-shape-plan-private.h"

/*XXX:should go in lib "global init"*/
static hb_shape_plan_t hb_shape_plan_nil = {
	REF_CNT_INVALID_VAL,
	TRUE,			/*default_shaper_list */
	NULL,			/*face */
	HB_SEGMENT_PROPERTIES_DEFAULT,	/*props */
	NULL,			/*shaper_func */
	NULL,			/*shaper_name */
	{
#ifdef HAVE_GRAPHITE2
	 HB_SHAPER_DATA_INVALID,
#endif
#ifdef HAVE_OT
	 HB_SHAPER_DATA_INVALID,
#endif
	 HB_SHAPER_DATA_INVALID	/*fallback */
	 }
};

/*TODO no user-feature caching for now.*/
struct hb_shape_plan_proposal_t {
	const hb_segment_properties_t props;
	const char *const *shaper_list;
	hb_shape_func_t *shaper_func;
};

hb_shape_plan_t *hb_shape_plan_get_empty(void)
{
	return &hb_shape_plan_nil;
}

#ifdef HAVE_GRAPHITE2
static INLINE hb_bool_t hb_graphite2_shaper_face_data_ensure(hb_face_t * face)
{
	while (1) {
		struct hb_graphite2_shaper_face_data_t *data;
		void *expected;

		data = hb_atomic_ptr_get(&face->shaper_data.graphite2);
		if (data)
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (!data)
			data = hb_graphite2_shaper_face_data_create(face);

		if (!data)
			data = HB_SHAPER_DATA_INVALID;

		expected = NULL;
		if (hb_atomic_ptr_cmpexch
		    (&face->shaper_data.graphite2, &expected, &data))
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (data != HB_SHAPER_DATA_INVALID
		    && data != HB_SHAPER_DATA_SUCCEEDED)
			hb_graphite2_shaper_face_data_destroy(data);
	}
}

static INLINE hb_bool_t hb_graphite2_shaper_font_data_ensure(hb_font_t * font)
{
	while (1) {
		struct hb_graphite2_shaper_font_data_t *data;
		void *expected;

		data = hb_atomic_ptr_get(&font->shaper_data.graphite2);
		if (data)
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (!data)
			data = hb_graphite2_shaper_font_data_create(font);

		if (!data)
			data = HB_SHAPER_DATA_INVALID;

		expected = NULL;
		if (hb_atomic_ptr_cmpexch
		    (&font->shaper_data.graphite2, &expected, &data))
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (data != HB_SHAPER_DATA_INVALID
		    && data != HB_SHAPER_DATA_SUCCEEDED)
			hb_graphite2_shaper_font_data_destroy(data);
	}
}
#endif

#ifdef HAVE_OT
static INLINE hb_bool_t hb_ot_shaper_face_data_ensure(hb_face_t * face)
{
	while (1) {
		struct hb_ot_shaper_face_data_t *data;
		void *expected;

		data = hb_atomic_ptr_get(&face->shaper_data.ot);
		if (data)
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (!data)
			data = hb_ot_shaper_face_data_create(face);

		if (!data)
			data = HB_SHAPER_DATA_INVALID;

		expected = NULL;
		if (hb_atomic_ptr_cmpexch
		    (&face->shaper_data.ot, &expected, &data))
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (data != HB_SHAPER_DATA_INVALID
		    && data != HB_SHAPER_DATA_SUCCEEDED)
			hb_ot_shaper_face_data_destroy(data);
	}
}

static INLINE hb_bool_t hb_ot_shaper_font_data_ensure(hb_font_t * font)
{
	while (1) {
		struct hb_ot_shaper_font_data_t *data;
		void *expected;

		data = hb_atomic_ptr_get(&font->shaper_data.ot);
		if (data)
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (!data)
			data = hb_ot_shaper_font_data_create(font);

		if (!data)
			data = HB_SHAPER_DATA_INVALID;

		expected = NULL;
		if (hb_atomic_ptr_cmpexch
		    (&font->shaper_data.ot, &expected, &data))
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (data != HB_SHAPER_DATA_INVALID
		    && data != HB_SHAPER_DATA_SUCCEEDED)
			hb_ot_shaper_font_data_destroy(data);
	}
}
#endif

static INLINE hb_bool_t hb_fallback_shaper_face_data_ensure(hb_face_t * face)
{
	while (1) {
		struct hb_fallback_shaper_face_data_t *data;
		void *expected;

		data = hb_atomic_ptr_get(&face->shaper_data.fallback);
		if (data)
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (!data)
			data = hb_fallback_shaper_face_data_create(face);

		if (!data)
			data = HB_SHAPER_DATA_INVALID;

		expected = NULL;
		if (hb_atomic_ptr_cmpexch
		    (&face->shaper_data.fallback, &expected, &data))
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (data != HB_SHAPER_DATA_INVALID
		    && data != HB_SHAPER_DATA_SUCCEEDED)
			hb_fallback_shaper_face_data_destroy(data);
	}
}

static INLINE hb_bool_t hb_fallback_shaper_font_data_ensure(hb_font_t * font)
{
	while (1) {
		struct hb_fallback_shaper_font_data_t *data;
		void *expected;

		data = hb_atomic_ptr_get(&font->shaper_data.fallback);
		if (data)
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (!data)
			data = hb_fallback_shaper_font_data_create(font);

		if (!data)
			data = HB_SHAPER_DATA_INVALID;

		expected = NULL;
		if (hb_atomic_ptr_cmpexch
		    (&font->shaper_data.fallback, &expected, &data))
			return !HB_SHAPER_DATA_IS_INVALID(data);

		if (data != HB_SHAPER_DATA_INVALID
		    && data != HB_SHAPER_DATA_SUCCEEDED)
			hb_fallback_shaper_font_data_destroy(data);
	}
}

static void hb_shape_plan_plan(hb_shape_plan_t * shape_plan,
			       const hb_feature_t * user_features,
			       unsigned num_user_features,
			       const char *const *shaper_list)
{
	struct hb_shaper_pair_t *shapers;
	unsigned i;

	shapers = hb_shapers_get();
	if (!shaper_list) {
		for (i = 0; i < HB_SHAPERS_COUNT; ++i)
			if (0) ;
#ifdef HAVE_GRAPHITE2
			else if (shapers[i].func == hb_graphite2_shape) {
				if (hb_graphite2_shaper_face_data_ensure
				    (shape_plan->face)) {
					shape_plan->shaper_data.graphite2 =
					    hb_graphite2_shaper_shape_plan_data_create
					    (shape_plan, user_features,
					     num_user_features);
					shape_plan->shaper_func =
					    hb_graphite2_shape;
					shape_plan->shaper_name = "graphite2";
					return;
				}
			}
#endif
#ifdef HAVE_OT
			else if (shapers[i].func == hb_ot_shape) {
				if (hb_ot_shaper_face_data_ensure
				    (shape_plan->face)) {
					shape_plan->shaper_data.ot =
					    hb_ot_shaper_shape_plan_data_create
					    (shape_plan, user_features,
					     num_user_features);
					shape_plan->shaper_func = hb_ot_shape;
					shape_plan->shaper_name = "ot";
					return;
				}
			}
#endif
			else if (shapers[i].func == hb_fallback_shape) {
				if (hb_fallback_shaper_face_data_ensure
				    (shape_plan->face)) {
					shape_plan->shaper_data.fallback =
					    hb_fallback_shaper_shape_plan_data_create
					    (shape_plan, user_features,
					     num_user_features);
					shape_plan->shaper_func =
					    hb_fallback_shape;
					shape_plan->shaper_name = "fallback";
					return;
				}
			}
	} else {
		for (; *shaper_list; ++shaper_list)
			if (0) ;
#ifdef HAVE_GRAPHITE2
			else if (0 == strcmp(*shaper_list, "graphite2")) {
				if (hb_graphite2_shaper_face_data_ensure
				    (shape_plan->face)) {
					shape_plan->shaper_data.graphite2 =
					    hb_graphite2_shaper_shape_plan_data_create
					    (shape_plan, user_features,
					     num_user_features);
					shape_plan->shaper_func =
					    hb_graphite2_shape;
					shape_plan->shaper_name = "graphite2";
					return;
				}
			}
#endif
#ifdef HAVE_OT
			else if (0 == strcmp(*shaper_list, "ot")) {
				if (hb_ot_shaper_face_data_ensure
				    (shape_plan->face)) {
					shape_plan->shaper_data.ot =
					    hb_ot_shaper_shape_plan_data_create
					    (shape_plan, user_features,
					     num_user_features);
					shape_plan->shaper_func = hb_ot_shape;
					shape_plan->shaper_name = "ot";
					return;
				}
			}
#endif
			else if (0 == strcmp(*shaper_list, "fallback")) {
				if (hb_fallback_shaper_face_data_ensure
				    (shape_plan->face)) {
					shape_plan->shaper_data.fallback =
					    hb_fallback_shaper_shape_plan_data_create
					    (shape_plan, user_features,
					     num_user_features);
					shape_plan->shaper_func =
					    hb_fallback_shape;
					shape_plan->shaper_name = "fallback";
					return;
				}
			}
	}
}

hb_shape_plan_t *hb_shape_plan_create(hb_face_t * face,
				      const hb_segment_properties_t * props,
				      const hb_feature_t * user_features,
				      unsigned int num_user_features,
				      const char *const *shaper_list)
{
	hb_shape_plan_t *shape_plan;

	assert(props->direction != HB_DIRECTION_INVALID);

	if (!face)
		face = hb_face_get_empty();

	if (!props
	    || hb_atomic_int32_get(&face->ref_cnt) == REF_CNT_INVALID_VAL)
		return hb_shape_plan_get_empty();

	shape_plan = calloc(1, sizeof(*shape_plan));
	if (!shape_plan)
		return hb_shape_plan_get_empty();
	hb_atomic_int32_set(&shape_plan->ref_cnt, 1);

	shape_plan->default_shaper_list = shaper_list == NULL;
	hb_face_make_immutable(face);
	shape_plan->face = hb_face_reference(face);
	shape_plan->props = *props;

	hb_shape_plan_plan(shape_plan, user_features, num_user_features,
			   shaper_list);
	return shape_plan;
}

static hb_bool_t hb_shape_plan_matches(hb_shape_plan_t * shape_plan,
				       struct hb_shape_plan_proposal_t
				       *proposal)
{
	return hb_segment_properties_equal(&shape_plan->props, &proposal->props)
	    &&
	    ((shape_plan->default_shaper_list && proposal->shaper_list == NULL)
	     || (shape_plan->shaper_func == proposal->shaper_func));
}

hb_shape_plan_t *hb_shape_plan_reference(hb_shape_plan_t * shape_plan)
{
	if (hb_atomic_int32_get(&shape_plan->ref_cnt) != REF_CNT_INVALID_VAL)
		hb_atomic_int32_add(&shape_plan->ref_cnt, 1);
	return shape_plan;
}

void hb_shape_plan_destroy(hb_shape_plan_t * shape_plan)
{
	if (!shape_plan)
		return;
	if (hb_atomic_int32_get(&shape_plan->ref_cnt) == REF_CNT_INVALID_VAL)
		return;
	hb_atomic_int32_add(&shape_plan->ref_cnt, -1);
	if (hb_atomic_int32_get(&shape_plan->ref_cnt) > 0)
		return;
	hb_atomic_int32_set(&shape_plan->ref_cnt, REF_CNT_INVALID_VAL);

#ifdef HAVE_GRAPHITE2
	if (shape_plan->shaper_data.graphite2
	    && shape_plan->shaper_data.graphite2 != HB_SHAPER_DATA_INVALID
	    && shape_plan->shaper_data.graphite2 != HB_SHAPER_DATA_SUCCEEDED)
		hb_graphite2_shaper_shape_plan_data_destroy
		    (shape_plan->shaper_data.graphite2);
#endif
#ifdef HAVE_OT
	if (shape_plan->shaper_data.ot
	    && shape_plan->shaper_data.ot != HB_SHAPER_DATA_INVALID
	    && shape_plan->shaper_data.ot != HB_SHAPER_DATA_SUCCEEDED)
		hb_ot_shaper_shape_plan_data_destroy(shape_plan->shaper_data.
						     ot);
#endif
	if (shape_plan->shaper_data.fallback
	    && shape_plan->shaper_data.fallback != HB_SHAPER_DATA_INVALID
	    && shape_plan->shaper_data.fallback != HB_SHAPER_DATA_SUCCEEDED)
		hb_fallback_shaper_shape_plan_data_destroy
		    (shape_plan->shaper_data.fallback);

	hb_face_destroy(shape_plan->face);
	free(shape_plan);
}

hb_shape_plan_t *hb_shape_plan_create_cached(hb_face_t * face,
					     const hb_segment_properties_t *
					     props,
					     const hb_feature_t * user_features,
					     unsigned int num_user_features,
					     const char *const *shaper_list)
{
	struct hb_shape_plan_proposal_t proposal;
	const char *const *shaper_item;
	struct plan_node_t *cached_plan_nodes;
	hb_shape_plan_t *shape_plan;

	if (num_user_features)
		return hb_shape_plan_create(face, props, user_features,
					    num_user_features, shaper_list);

	/*XXX:proposal.props = *props;*/
	memcpy((void*)&proposal.props,(void*)props,sizeof(proposal.props));
	proposal.shaper_list = shaper_list;
	proposal.shaper_func = NULL;

	if (shaper_list) {
		/*Choose shaper.  Adapted from hb_shape_plan_plan(). */
		for (shaper_item = shaper_list; *shaper_item; shaper_item++)
			if (0) ;
#ifdef HAVE_GRAPHITE2
			else if (0 == strcmp(*shaper_item, "graphite2")) {
				if (hb_graphite2_shaper_face_data_ensure(face))
					proposal.shaper_func =
					    hb_graphite2_shape;
			}
#endif
#ifdef HAVE_OT
			else if (0 == strcmp(*shaper_item, "ot")) {
				if (hb_ot_shaper_face_data_ensure(face))
					proposal.shaper_func = hb_ot_shape;
			}
#endif
			else if (0 == strcmp(*shaper_item, "fallback")) {
				if (hb_fallback_shaper_face_data_ensure(face))
					proposal.shaper_func =
					    hb_fallback_shape;
			}

		if (!proposal.shaper_list)
			return hb_shape_plan_get_empty();
	}

	cached_plan_nodes = hb_atomic_ptr_get(&face->shape_plans);
	while (1) {
		struct plan_node_t *node;

		for (node = cached_plan_nodes; node; node = node->next)
			if (hb_shape_plan_matches(node->shape_plan, &proposal))
				return
				    hb_shape_plan_reference(node->shape_plan);

		/*Not found. */

		shape_plan =
		    hb_shape_plan_create(face, props, user_features,
					 num_user_features, shaper_list);
		node = calloc(1, sizeof(*node));
		if (!node)
			return shape_plan;

		node->shape_plan = shape_plan;
		node->next = cached_plan_nodes;

		if (hb_atomic_ptr_cmpexch
		    (&face->shape_plans, &cached_plan_nodes, &node))
			break;

		hb_shape_plan_destroy(shape_plan);
		free(node);
	}

	/*Release our reference on face. */
	hb_face_destroy(face);
	return hb_shape_plan_reference(shape_plan);
}

hb_bool_t hb_shape_plan_execute(hb_shape_plan_t * shape_plan, hb_font_t * font,
				hb_buffer_t * buffer,
				const hb_feature_t * features,
				unsigned num_features)
{
	if (hb_atomic_int32_get(&shape_plan->ref_cnt) == REF_CNT_INVALID_VAL
	    || hb_atomic_int32_get(&font->ref_cnt) == REF_CNT_INVALID_VAL
	    || hb_atomic_int32_get(&buffer->ref_cnt) == REF_CNT_INVALID_VAL)
		return FALSE;

	assert(shape_plan->face == font->face);
	assert(hb_segment_properties_equal(&shape_plan->props, &buffer->props));

	if (0) ;
#ifdef HAVE_GRAPHITE2
	else if (shape_plan->shaper_func == hb_graphite2_shape)
		return shape_plan->shaper_data.graphite2
		    && hb_graphite2_shaper_font_data_ensure(font)
		    && hb_graphite2_shape(shape_plan, font, buffer, features,
					  num_features);
#endif
#ifdef HAVE_OT
	else if (shape_plan->shaper_func == hb_ot_shape)
		return shape_plan->shaper_data.ot
		    && hb_ot_shaper_font_data_ensure(font)
		    && hb_ot_shape(shape_plan, font, buffer, features,
				   num_features);
#endif
	else if (shape_plan->shaper_func == hb_fallback_shape)
		return shape_plan->shaper_data.fallback
		    && hb_fallback_shaper_font_data_ensure(font)
		    && hb_fallback_shape(shape_plan, font, buffer, features,
					 num_features);
	return FALSE;
}


Mode Type Size Ref File
100644 blob 215 9381999f3aeff23f92475e7aef7a141e7b31c938 ABBREVIATIONS
100644 blob 120 b5b8344f6c0ab2a879570909dfc67753d036ea10 FIXME
100644 blob 241 9c28271a27b1d0acd9a2a8f412a4f107767e4df2 README
100644 blob 219 bc171ca45d9157358b008496850fe8f73dd0c66b harfbuzz.pc.in
100644 blob 1192 833e0c5e348c3c7663711b84cd05baf7f2ec4b6b hb-atomic-private.h
100644 blob 119 154fbbb0957b9a8247993436984ab9cda943720c hb-blob-private.h
100644 blob 4603 6cc79394dfa9884d604a6e958268d841c370f3f1 hb-blob.c
100644 blob 3534 ef3fc98c05de6d2fa43f6303d6419f6620d474bf hb-blob.h
100644 blob 1326 3b93b65480793ced5bdd1c6e2587d3c2a66f3a53 hb-buffer-private.h
100644 blob 11355 24118c098a5412ec39226fe2a18b7e192dd902b2 hb-buffer.c
100644 blob 10513 d3a2512d800fde857f72764b150aaa392fd8b15d hb-buffer.h
100644 blob 3302 92b54dbaf82493e6f281bb20231df3e1ecb9d63c hb-common.c
100644 blob 13612 af17d780c27fc757441d5ce0fce1db120bd87c9b hb-common.h
100644 blob 1746 82066e4e0dc0b2770a8800f9a9fb417e14d9d7d0 hb-coretext.h
100644 blob 1584 30ae4b1caf76dd03cc682eda7458728b5380ad6c hb-deprecated.h
100644 blob 1175 5e43e30983efc3b0936efbdaf3921eecbfc1635a hb-face-private.h
100644 blob 5222 770d81b8c6b9bafd4c9eb018f1420e369ba09647 hb-face.c
100644 blob 2985 91237b70850bc9750516c73aca7051228e5c22c1 hb-face.h
100644 blob 3268 cdc0ae9a6a50a1b444db3a148db08f01bececa7c hb-fallback-shape.c
100644 blob 2875 dce2474d44b59f5349528718422dd9af7975cf52 hb-font-private.h
100644 blob 24805 80df40b5a885a7e2ae668ef1fc01020247a4e155 hb-font.c
100644 blob 13844 c3165e1d74c123fc1aa289ce503e38f870b43611 hb-font.h
100644 blob 2502 551d6e47f67a13ce27d21f4bfadc86a8091507ce hb-ft.c
100644 blob 3790 dc8ef8558484cb4faaa3cbe6906a84d8005a3a10 hb-ft.h
100644 blob 3637 1468f247a5f2d3c48a0de2a9a9156e17fe4521dc hb-glib.c
100644 blob 1627 a6b6ff82026f3a0016eab1f5201e13afbde93feb hb-glib.h
100644 blob 3140 0ea3b12cf1676dc6a2da8e546d9d6b7cbc60ed41 hb-gobject-structs.h
100644 blob 1334 ea1bd25df82d46a222436a1ad078d0fc95d5c303 hb-gobject.h
100644 blob 1461 122c3e476366b341bc74ced282f424d863255ca7 hb-graphite2.h
100644 blob 1525 2db6a7b6797bddc46688f7b583de991a956a2e2a hb-icu.h
100644 blob 2133 3e4514517efb14434d2ac3a08b82a06eedf8ffaa hb-open-file-private.h
100644 blob 2674 18b89705a885521c6d8d08448eac1f65395a3a0d hb-open-file.c
100644 blob 65 642c3d3d45fdc42619d000cc9f7b3c890bdc0190 hb-ot-face-private.h
100644 blob 1365 80eaa54b1add7333d975fbee27c00bc777f3e046 hb-ot-font.h
100644 blob 2784 91122b51fbb221aeb52634439405e47e700a1e80 hb-ot-layout.c
100644 blob 9658 eb23d45b68aabd24538b46bb482ac187c5f61d73 hb-ot-layout.h
100644 blob 1712 7b1bcc0637808bbbceda020409b61d947237d925 hb-ot-shape.h
100644 blob 730 bfa3d45093e12bf9aa7a0a8920d6a573145e1f99 hb-ot-tag.c
100644 blob 1728 54fb747f582301c6d46bbe33ede8a2ef127dc999 hb-ot-tag.h
100644 blob 1348 47c92a58e4b7ddc3225e7cb65ce20aafceb3b81a hb-ot.h
100644 blob 2541 827e6c9b4e8f90ba02b24a259ddd10fa8b7f34cb hb-private.h
100644 blob 3828 2164c1a6549afc39d9f3c6e82bb6ca8f0b8f732d hb-set.h
100644 blob 1417 62ff962974ac05946c099ea7d4fb3473bd95f6ba hb-shape-plan-private.h
100644 blob 14266 11f405cdb548fd9ac6aaebda60059202512d5831 hb-shape-plan.c
100644 blob 2837 aa5e0c7d6f519b841c4519811f18845cdf92ee77 hb-shape-plan.h
100644 blob 5057 00628cd3079786a1cf9c2534458ceeac98bcc9b6 hb-shape.c
100644 blob 2170 53bb845bf48f40758c69ae31a399054b43690413 hb-shape.h
100644 blob 1015 ee80e4526b63481a48b348fe279c15e757363776 hb-shaper-private.h
100644 blob 2250 8073a7cc168f5624424250b6a81f0440243fc015 hb-shaper.c
100644 blob 1210 276c97e391a35215a71ad538b77696c69dd16eb1 hb-unicode-private.h
100644 blob 7644 53862e6db7e90afb058efbc0f0bad27cd25b28c1 hb-unicode.c
100644 blob 14695 33b68aa0d85b666090c2820e46c7a5242992b228 hb-unicode.h
100644 blob 1391 4e4ef9986a62e3df91d1d4ea048ec89d3526498d hb-uniscribe.h
100644 blob 4990 319e112fe2de8d3ccb9249fec39a96971d13f1fc hb-utf-private.c
100644 blob 1246 dca4a18b95114b239706d6917a6c74eac87b3279 hb-utf-private.h
100644 blob 1887 0ffd889b278962a33f2b8882e1dba4b41af6b672 hb-version.h.in
100644 blob 1521 7402034f4379372b3a7d0529142e4c87e4b4599d hb.h
100644 blob 7651 65c5ca88a67c30becee01c5a8816d964b03862f9 license.md
100755 blob 6112 017403a7717449e9a432c10a911340b211c5d974 make
Hints:
Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://rocketgit.com/user/sylware/charfbuzz

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@ssh.rocketgit.com/user/sylware/charfbuzz

Clone this repository using git:
git clone git://git.rocketgit.com/user/sylware/charfbuzz

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a merge request:
... clone the repository ...
... make some changes and some commits ...
git push origin main