#include "../features.c" #include "../memleak.c" #include #include #include #include #include #include /* AUTOGENERATED MOCKS START */ /* Generated stub for amount_asset_is_main */ bool amount_asset_is_main(struct amount_asset *asset UNNEEDED) { fprintf(stderr, "amount_asset_is_main called!\n"); abort(); } /* Generated stub for amount_asset_to_sat */ struct amount_sat amount_asset_to_sat(struct amount_asset *asset UNNEEDED) { fprintf(stderr, "amount_asset_to_sat called!\n"); abort(); } /* Generated stub for amount_sat_add */ bool amount_sat_add(struct amount_sat *val UNNEEDED, struct amount_sat a UNNEEDED, struct amount_sat b UNNEEDED) { fprintf(stderr, "amount_sat_add called!\n"); abort(); } /* Generated stub for amount_sat_eq */ bool amount_sat_eq(struct amount_sat a UNNEEDED, struct amount_sat b UNNEEDED) { fprintf(stderr, "amount_sat_eq called!\n"); abort(); } /* Generated stub for amount_sat_sub */ bool amount_sat_sub(struct amount_sat *val UNNEEDED, struct amount_sat a UNNEEDED, struct amount_sat b UNNEEDED) { fprintf(stderr, "amount_sat_sub called!\n"); abort(); } /* Generated stub for fromwire_fail */ const void *fromwire_fail(const u8 **cursor UNNEEDED, size_t *max UNNEEDED) { fprintf(stderr, "fromwire_fail called!\n"); abort(); } /* Generated stub for fromwire_tal_arrn */ u8 *fromwire_tal_arrn(const tal_t *ctx UNNEEDED, const u8 **cursor UNNEEDED, size_t *max UNNEEDED, size_t num UNNEEDED) { fprintf(stderr, "fromwire_tal_arrn called!\n"); abort(); } /* Generated stub for fromwire_u16 */ u16 fromwire_u16(const u8 **cursor UNNEEDED, size_t *max UNNEEDED) { fprintf(stderr, "fromwire_u16 called!\n"); abort(); } /* Generated stub for towire_u16 */ void towire_u16(u8 **pptr UNNEEDED, u16 v UNNEEDED) { fprintf(stderr, "towire_u16 called!\n"); abort(); } /* Generated stub for towire_u8_array */ void towire_u8_array(u8 **pptr UNNEEDED, const u8 *arr UNNEEDED, size_t num UNNEEDED) { fprintf(stderr, "towire_u8_array called!\n"); abort(); } /* AUTOGENERATED MOCKS END */ static void test_featurebits_or(void) { u8 *f1 = tal_arr(tmpctx, u8, 0); u8 *f2 = tal_arr(tmpctx, u8, 0); u8 *control = tal_arr(tmpctx, u8, 0); for (size_t i = 0; i < 100; i += 3) { set_feature_bit(&f1, i); set_feature_bit(&control, i); } for (size_t i = 0; i < 10; i += 2) { set_feature_bit(&f2, i); set_feature_bit(&control, i); } u8 *result = featurebits_or(tmpctx, take(f1), f2); assert( memeq(result, tal_bytelen(result), control, tal_bytelen(control))); } static void setup_features(void) { static const u32 default_features[] = { OPTIONAL_FEATURE(OPT_DATA_LOSS_PROTECT), OPTIONAL_FEATURE(OPT_UPFRONT_SHUTDOWN_SCRIPT), OPTIONAL_FEATURE(OPT_GOSSIP_QUERIES), OPTIONAL_FEATURE(OPT_VAR_ONION), OPTIONAL_FEATURE(OPT_PAYMENT_SECRET), OPTIONAL_FEATURE(OPT_BASIC_MPP), OPTIONAL_FEATURE(OPT_GOSSIP_QUERIES_EX), OPTIONAL_FEATURE(OPT_STATIC_REMOTEKEY), }; u8 *f = tal_arr(NULL, u8, 0); for (size_t i = 0; i < ARRAY_SIZE(default_features); i++) set_feature_bit(&f, default_features[i]); features_core_init(take(f)); } int main(void) { u8 *bits, *lf; setup_locale(); wally_init(0); secp256k1_ctx = wally_get_secp_context(); setup_tmpctx(); setup_features(); bits = tal_arr(tmpctx, u8, 0); for (size_t i = 0; i < 100; i += 3) set_feature_bit(&bits, i); for (size_t i = 0; i < 100; i++) assert(test_bit(bits, i / 8, i % 8) == ((i % 3) == 0)); for (size_t i = 0; i < 100; i++) assert(feature_is_set(bits, i) == ((i % 3) == 0)); /* Simple test: single byte */ bits = tal_arr(tmpctx, u8, 1); /* Compulsory feature */ bits[0] = (1 << 0); assert(feature_offered(bits, 0)); assert(!feature_offered(bits, 2)); assert(!feature_offered(bits, 8)); assert(!feature_offered(bits, 16)); /* Optional feature */ bits[0] = (1 << 1); assert(feature_offered(bits, 0)); assert(!feature_offered(bits, 2)); assert(!feature_offered(bits, 8)); assert(!feature_offered(bits, 16)); /* Endian-sensitive test: big-endian means we frob last byte here */ bits = tal_arrz(tmpctx, u8, 2); bits[1] = (1 << 0); assert(feature_offered(bits, 0)); assert(!feature_offered(bits, 2)); assert(!feature_offered(bits, 8)); assert(!feature_offered(bits, 16)); /* Optional feature */ bits[1] = (1 << 1); assert(feature_offered(bits, 0)); assert(!feature_offered(bits, 2)); assert(!feature_offered(bits, 8)); assert(!feature_offered(bits, 16)); /* We always support no features. */ memset(bits, 0, tal_count(bits)); assert(features_unsupported(bits) == -1); /* We must support our own features. */ lf = get_offered_initfeatures(tmpctx); assert(features_unsupported(lf) == -1); /* We can add random odd features, no problem. */ for (size_t i = 1; i < 16; i += 2) { bits = tal_dup_talarr(tmpctx, u8, lf); set_feature_bit(&bits, i); assert(features_unsupported(bits) == -1); } /* We can't add random even features. */ for (size_t i = 0; i < 16; i += 2) { bits = tal_dup_talarr(tmpctx, u8, lf); set_feature_bit(&bits, i); /* Special case for missing compulsory feature */ if (i == 2) { assert(features_unsupported(bits) == i); } else { assert((features_unsupported(bits) == -1) == feature_offered(our_features->bits[INIT_FEATURE], i)); } } test_featurebits_or(); wally_cleanup(0); tal_free(tmpctx); take_cleanup(); features_cleanup(); return 0; }