From dd9390c583e884f3b09d8a9f6f36c569f3faacea Mon Sep 17 00:00:00 2001 From: "Jay Freeman (saurik)" Date: Mon, 20 Apr 2009 00:32:33 +0000 Subject: [PATCH 1/1] Localization fixes and performance improvements. --- Cydia.app/English.lproj/Localizable.strings | 8 +- Cydia.app/French.lproj/Localizable.strings | 8 +- Cydia.app/German.lproj/Localizable.strings | 8 +- Cydia.mm | 185 ++++++++++++++------ 4 files changed, 141 insertions(+), 68 deletions(-) diff --git a/Cydia.app/English.lproj/Localizable.strings b/Cydia.app/English.lproj/Localizable.strings index 9d990332..01b1a77c 100644 --- a/Cydia.app/English.lproj/Localizable.strings +++ b/Cydia.app/English.lproj/Localizable.strings @@ -45,7 +45,7 @@ "DISK_USING" = "Disk Using"; "DONATE_USING" = "Donate using %@"; "DONE" = "Done"; -"DOUBLE_QUOTED": "“$@”"; +"DOUBLE_QUOTED" = "“%@”"; "DOWNGRADE" = "Downgrade"; "DOWNLOADING" = "Downloading"; "EDIT" = "Edit"; @@ -55,7 +55,7 @@ "ESSENTIAL_UPGRADE" = "Essential Upgrade"; "ESSENTIAL_UPGRADES" = "%d Essential Upgrades"; "ESSENTIAL_UPGRADE_EX" = "One or more essential packages are currently out of date. If these upgrades are not performed you are likely to encounter errors."; -"EXCLAMATION": "%@!"; +"EXCLAMATION" = "%@!"; "EXPANDED_SIZE" = "Expanded Size"; "EXPERT" = "Expert"; "FEATURED_PACKAGES" = "Featured Packages"; @@ -123,7 +123,7 @@ "PURGE" = "Purge"; "Q_D" = "Q'd"; "QUEUED_FOR" = "Queued for %@"; -"QUESTION": "%@?"; +"QUESTION" = "%@?"; "REBOOT_DEVICE" = "Reboot Device"; "RECENT_CHANGES_UPDATES" = "Recent Changes/Updates"; "RECOMMENDED_BOOKS" = "Recommended Books"; @@ -160,7 +160,7 @@ "SHOW_ALL_CHANGES_EX" = "Changes only shows upgrades to installed packages so as to minimize spam from packagers. Activate this to see upgrades to this package even when it is not installed."; "SIGNED_PACKAGE" = "This package has been signed."; "SIMPLE" = "Simple"; -"SINGLE_QUOTED": "‘$@’"; +"SINGLE_QUOTED" = "‘%@’"; "SLASH_DELIMITED" = "%@ / %@"; "SOURCE_DETAILS" = "Source Information"; "SOURCE_WARNING" = "Source Warning"; diff --git a/Cydia.app/French.lproj/Localizable.strings b/Cydia.app/French.lproj/Localizable.strings index e2752569..14149426 100644 --- a/Cydia.app/French.lproj/Localizable.strings +++ b/Cydia.app/French.lproj/Localizable.strings @@ -45,7 +45,7 @@ "DISK_USING" = "Espace Disque Utilisé"; "DONATE_USING" = "Donnez en utilisant %@"; "DONE" = "OK"; -"DOUBLE_QUOTED": "« $@ »"; +"DOUBLE_QUOTED" = "« $@ »"; "DOWNGRADE" = "Rétrograder"; "DOWNLOADING" = "Téléchargement"; "EDIT" = "Éditer"; @@ -55,7 +55,7 @@ "ESSENTIAL_UPGRADE" = "Mise à Jour Essentielle"; "ESSENTIAL_UPGRADES" = "%d Mises à Jour Essentielles"; "ESSENTIAL_UPGRADE_EX" = "Un ou plusieurs paquets essentiels sont périmés. Si vous ne les mettez pas à jour, vous pourriez rencontrer des erreurs."; -"EXCLAMATION": "%@ !"; +"EXCLAMATION" = "%@ !"; "EXPANDED_SIZE" = "Taille Décompressé"; "EXPERT" = "Expert"; "FEATURED_PACKAGES" = "Paquets Conseillés"; @@ -123,7 +123,7 @@ "PURGE" = "Purger"; "Q_D" = "Q'd"; "QUEUED_FOR" = "File d'Attente pour %@"; -"QUESTION": "%@ ?"; +"QUESTION" = "%@ ?"; "REBOOT_DEVICE" = "Redémarrer l'Appareil"; "RECENT_CHANGES_UPDATES" = "Mises à Jour ou Changements Récents"; "RECOMMENDED_BOOKS" = "Livres Recommandés"; @@ -160,7 +160,7 @@ "SHOW_ALL_CHANGES_EX" = "Seuls les mises à jour de paquets installés ne sont affichés ici pour limiter le spam des empaqueteurs. En activant ceci vous serez averti des mises à jour sans que ce paquet soit installé."; "SIGNED_PACKAGE" = "Ce Paquet est Signé."; "SIMPLE" = "Simple"; -"SINGLE_QUOTED": "‹ $@ ›"; +"SINGLE_QUOTED" = "‹ %@ ›"; "SLASH_DELIMITED" = "%@ / %@"; "SOURCE_DETAILS" = "Détails sur la Source"; "SOURCE_WARNING" = "Alerte Source"; diff --git a/Cydia.app/German.lproj/Localizable.strings b/Cydia.app/German.lproj/Localizable.strings index 36157112..13e119e0 100644 --- a/Cydia.app/German.lproj/Localizable.strings +++ b/Cydia.app/German.lproj/Localizable.strings @@ -45,7 +45,7 @@ "DISK_USING" = "Speicherplatz-Verbrauch"; "DONATE_USING" = "Spenden via %@"; "DONE" = "Fertig"; -"DOUBLE_QUOTED": "„$@“"; +"DOUBLE_QUOTED" = "„%@“"; "DOWNGRADE" = "Downgrade"; "DOWNLOADING" = "Herunterladen"; "EDIT" = "Bearbeiten"; @@ -55,7 +55,7 @@ "ESSENTIAL_UPGRADE" = "Wichtige Aktualisierung"; "ESSENTIAL_UPGRADES" = "%d wichtige Aktualisierungen"; "ESSENTIAL_UPGRADE_EX" = "Ein oder mehr wichtige Pakete sind veraltet, was zu Fehlern führen kann. Es wird empfohlen die Aktualisierungen zu installieren."; -"EXCLAMATION": "%@!"; +"EXCLAMATION" = "%@!"; "EXPANDED_SIZE" = "Entpackte Größe"; "EXPERT" = "Experte"; "FEATURED_PACKAGES" = "Vorgeschlagene Pakete"; @@ -123,7 +123,7 @@ "PURGE" = "Leeren"; "Q_D" = "S"; "QUEUED_FOR" = "In Warteschlange für %@"; -"QUESTION": "%@?"; +"QUESTION" = "%@?"; "REBOOT_DEVICE" = "Gerät neustarten"; "RECENT_CHANGES_UPDATES" = "Neueste Änderungen/Aktualisierungen"; "RECOMMENDED_BOOKS" = "Empfohlene Bücher"; @@ -160,7 +160,7 @@ "SHOW_ALL_CHANGES_EX" = "Änderungen zeigt nur Aktualisierungen zu installierten Paketen an um Spam von Paket-Herstellern zu reduzieren. Aktivieren Sie dies, um Aktualisierungen zu diesem Paket zu sehen, obwohl es nicht installiert ist."; "SIGNED_PACKAGE" = "Dieses Paket wurde signiert."; "SIMPLE" = "Einfach"; -"SINGLE_QUOTED": "‚$@‘"; +"SINGLE_QUOTED" = "‚%@‘"; "SLASH_DELIMITED" = "%@ / %@"; "SOURCE_DETAILS" = "Information zur Quelle"; "SOURCE_WARNING" = "Quellen Warnung"; diff --git a/Cydia.mm b/Cydia.mm index 60067ebe..2f299b6c 100644 --- a/Cydia.mm +++ b/Cydia.mm @@ -417,9 +417,11 @@ extern NSString * const kCAFilterNearest; #endif /* Radix Sort {{{ */ +typedef uint32_t (*SKRadixFunction)(id, void *); + @interface NSMutableArray (Radix) - (void) radixSortUsingSelector:(SEL)selector withObject:(id)object; -- (void) radixSortUsingFunction:(uint32_t (*)(id, void *))function withArgument:(void *)argument; +- (void) radixSortUsingFunction:(SKRadixFunction)function withContext:(void *)argument; @end struct RadixItem_ { @@ -515,7 +517,7 @@ static void RadixSort_(NSMutableArray *self, size_t count, struct RadixItem_ *sw RadixSort_(self, count, swap); } -- (void) radixSortUsingFunction:(uint32_t (*)(id, void *))function withArgument:(void *)argument { +- (void) radixSortUsingFunction:(SKRadixFunction)function withContext:(void *)argument { size_t count([self count]); struct RadixItem_ *swap(new RadixItem_[count * 2]); @@ -534,6 +536,19 @@ static void RadixSort_(NSMutableArray *self, size_t count, struct RadixItem_ *sw /* }}} */ /* Insertion Sort {{{ */ +CFIndex SKBSearch_(const void *element, CFIndex elementSize, const void *list, CFIndex count, CFComparatorFunction comparator, void *context) { + const char *ptr = (const char *)list; + while (0 < count) { + CFIndex half = count / 2; + const char *probe = ptr + elementSize * half; + CFComparisonResult cr = comparator(element, probe, context); + if (0 == cr) return (probe - (const char *)list) / elementSize; + ptr = (cr < 0) ? ptr : probe + elementSize; + count = (cr < 0) ? half : (half + (count & 1) - 1); + } + return (ptr - (const char *)list) / elementSize; +} + CFIndex CFBSearch_(const void *element, CFIndex elementSize, const void *list, CFIndex count, CFComparatorFunction comparator, void *context) { const char *ptr = (const char *)list; while (0 < count) { @@ -547,24 +562,48 @@ CFIndex CFBSearch_(const void *element, CFIndex elementSize, const void *list, C return (ptr - (const char *)list) / elementSize; } +#define HistogramInsertionSort 1 + void CFArrayInsertionSortValues(CFMutableArrayRef array, CFRange range, CFComparatorFunction comparator, void *context) { if (range.length == 0) return; const void **values(new const void *[range.length]); CFArrayGetValues(array, range, values); +#if HistogramInsertionSort + uint32_t total(0), *offsets(new uint32_t[range.length]); +#endif + for (CFIndex index(1); index != range.length; ++index) { const void *value(values[index]); - CFIndex correct(CFBSearch_(&value, sizeof(const void *), values, index, comparator, context)); - //NSLog(@"%u %u", index, correct); + //CFIndex correct(SKBSearch_(&value, sizeof(const void *), values, index, comparator, context)); + CFIndex correct(index); + while (comparator(value, values[correct - 1], context) == kCFCompareLessThan) + if (--correct == 0) + break; if (correct != index) { - memmove(values + correct + 1, values + correct, sizeof(const void *) * (index - correct)); + size_t offset(index - correct); +#if HistogramInsertionSort + total += offset; + ++offsets[offset]; + if (offset > 10) + NSLog(@"Heavy Insertion Displacement: %u = %@", offset, value); +#endif + memmove(values + correct + 1, values + correct, sizeof(const void *) * offset); values[correct] = value; } } CFArrayReplaceValues(array, range, values, range.length); delete [] values; + +#if HistogramInsertionSort + for (CFIndex index(0); index != range.length; ++index) + if (offsets[index] != 0) + NSLog(@"Insertion Displacement [%u]: %u", index, offsets[index]); + NSLog(@"Average Insertion Displacement: %f", double(total) / range.length); + delete [] offsets; +#endif } /* }}} */ @@ -783,12 +822,16 @@ class CYString { return size_ == rhs.size_ && memcmp(data_, rhs.data_, size_) == 0; } - operator id() { + operator CFStringRef() { if (cache_ == NULL) { if (size_ == 0) return nil; cache_ = CFStringCreateWithBytesNoCopy(kCFAllocatorDefault, reinterpret_cast(data_), size_, kCFStringEncodingUTF8, NO, kCFAllocatorNull); - } return (id) cache_; + } return cache_; + } + + _finline operator id() { + return (NSString *) static_cast(*this); } }; @@ -1132,10 +1175,6 @@ NSString *Simplify(NSString *title) { return title; } - -_finline static void Stifle(char &value) { - value = (value & 0xdf) ^ 0x40; -} /* }}} */ bool isSectionVisible(NSString *section) { @@ -1721,7 +1760,8 @@ class Progress : - (NSArray *) purposes; - (bool) isCommercial; -- (uint32_t) compareByPrefix; +- (CYString &) cyname; + - (uint32_t) compareBySection:(NSArray *)sections; - (uint32_t) compareForChanges; @@ -1763,27 +1803,73 @@ uint32_t PackageChangesRadix(Package *self, void *) { return _not(uint32_t) - value.key; } -CFStringRef (*PackageName)(Package *self, SEL sel); +_finline static void Stifle(uint8_t &value) { +} -CFComparisonResult PackageNameCompare(Package *lhs, Package *rhs, void *arg) { - _profile(PackageNameCompare) - CFStringRef lhn, rhn; - CFIndex length; +uint32_t PackagePrefixRadix(Package *self, void *context) { + size_t offset(reinterpret_cast(context)); + CYString &name([self cyname]); - _profile(PackageNameCompare$Setup) - lhn = PackageName(lhs, @selector(name)); - rhn = PackageName(rhs, @selector(name)); - _end + size_t size(name.size()); + if (size == 0) + return 0; + char *text(name.data()); - _profile(PackageNameCompare$Nothing) - _end + size_t zeros; + if (!isdigit(text[0])) + zeros = 0; + else { + size_t digits(1); + while (size != digits && isdigit(text[digits])) + if (++digits == 4) + break; + zeros = 4 - digits; + } - _profile(PackageNameCompare$Length) - length = CFStringGetLength(lhn); - _end + uint8_t data[4]; + + // 0.607997 + + if (offset == 0 && zeros != 0) { + memset(data, '0', zeros); + memcpy(data + zeros, text, 4 - zeros); + } else { + /* XXX: there's some danger here if you request a non-zero offset < 4 and it gets zero padded */ + if (size <= offset - zeros) + return 0; + + text += offset - zeros; + size -= offset - zeros; + + if (size >= 4) + memcpy(data, text, 4); + else { + memcpy(data, text, size); + memset(data + size, 0, 4 - size); + } + + for (size_t i(0); i != 4; ++i) + if (isalpha(data[i])) + data[i] &= 0xdf; + } + + if (offset == 0) + data[0] = (data[0] & 0x3f) | "\x80\x00\xc0\x40"[data[0] >> 6]; + + /* XXX: ntohl may be more honest */ + return OSSwapInt32(*reinterpret_cast(data)); +} + +CYString &(*PackageName)(Package *self, SEL sel); + +CFComparisonResult PackageNameCompare(Package *lhs, Package *rhs, void *arg) { + _profile(PackageNameCompare) + CYString &lhi(PackageName(lhs, @selector(cyname))); + CYString &rhi(PackageName(rhs, @selector(cyname))); + CFStringRef lhn(lhi), rhn(rhi); _profile(PackageNameCompare$NumbersLast) - if (length != 0 && CFStringGetLength(rhn) != 0) { + if (!lhi.empty() && !rhi.empty()) { UniChar lhc(CFStringGetCharacterAtIndex(lhn, 0)); UniChar rhc(CFStringGetCharacterAtIndex(rhn, 0)); bool lha(CFUniCharIsMemberOf(lhc, kCFUniCharLetterCharacterSet)); @@ -1792,6 +1878,8 @@ CFComparisonResult PackageNameCompare(Package *lhs, Package *rhs, void *arg) { } _end + CFIndex length = CFStringGetLength(lhn); + _profile(PackageNameCompare$Compare) return CFStringCompareWithOptionsAndLocale(lhn, rhn, CFRangeMake(0, length), LaxCompareFlags_, Locale_); _end @@ -1812,6 +1900,10 @@ struct PackageNameOrdering : @implementation Package +- (NSString *) description { + return [NSString stringWithFormat:@"", static_cast(name_)]; +} + - (void) dealloc { if (source_ != nil) [source_ release]; @@ -2261,7 +2353,7 @@ struct PackageNameOrdering : NSString *section = [self simpleSection]; UIImage *icon(nil); - if (icon_ != nil) + if (!icon_.empty()) if ([icon_ hasPrefix:@"file:///"]) icon = [UIImage imageAtPath:[icon_ substringFromIndex:7]]; if (icon == nil) if (section != nil) @@ -2299,7 +2391,7 @@ struct PackageNameOrdering : } - (NSString *) support { - return support_ != nil ? support_ : [[self source] supportForPackage:id_]; + return !support_.empty() ? support_ : [[self source] supportForPackage:id_]; } - (NSArray *) files { @@ -2485,29 +2577,8 @@ struct PackageNameOrdering : return [self hasTag:@"cydia::commercial"]; } -- (uint32_t) compareByPrefix { - size_t offset(0); - - CYString &name(name_.empty() ? id_ : name_); - if (name.size() <= offset) - return 0; - size_t size(name.size() - offset); - - char data[4]; - if (size >= 4) - memcpy(data, name.data() + offset, 4); - else { - memcpy(data, name.data() + offset, size); - memset(data + size, 0, 4 - size); - } - - Stifle(data[0]); - Stifle(data[1]); - Stifle(data[2]); - Stifle(data[3]); - - /* XXX: ntohl may be more honest */ - return OSSwapInt32(*reinterpret_cast(data)); +- (CYString &) cyname { + return name_.empty() ? id_ : name_; } - (uint32_t) compareBySection:(NSArray *)sections { @@ -3111,7 +3182,9 @@ static NSArray *Finishes_; packages_ = [[NSArray alloc] initWithObjects:&packages.front() count:packages.size()]; _trace();*/ - [packages_ radixSortUsingSelector:@selector(compareByPrefix) withObject:NULL]; + [packages_ radixSortUsingFunction:reinterpret_cast(&PackagePrefixRadix) withContext:reinterpret_cast(16)]; + [packages_ radixSortUsingFunction:reinterpret_cast(&PackagePrefixRadix) withContext:reinterpret_cast(4)]; + [packages_ radixSortUsingFunction:reinterpret_cast(&PackagePrefixRadix) withContext:reinterpret_cast(0)]; /*_trace(); PrintTimes(); @@ -3125,7 +3198,7 @@ static NSArray *Finishes_; //CFArraySortValues((CFMutableArrayRef) packages_, CFRangeMake(0, [packages_ count]), reinterpret_cast(&PackageNameCompare), NULL); - CFArrayInsertionSortValues((CFMutableArrayRef) packages_, CFRangeMake(0, [packages_ count]), reinterpret_cast(&PackageNameCompare_), NULL); + CFArrayInsertionSortValues((CFMutableArrayRef) packages_, CFRangeMake(0, [packages_ count]), reinterpret_cast(&PackageNameCompare), NULL); //[packages_ sortUsingFunction:reinterpret_cast(&PackageNameCompare) context:NULL]; @@ -6476,7 +6549,7 @@ bool DepSubstrate(const pkgCache::VerIterator &iterator) { [packages_ addObject:package]; _trace(); - [packages_ radixSortUsingFunction:reinterpret_cast(&PackageChangesRadix) withArgument:NULL]; + [packages_ radixSortUsingFunction:reinterpret_cast(&PackageChangesRadix) withContext:NULL]; _trace(); Section *upgradable = [[[Section alloc] initWithName:CYLocalize("AVAILABLE_UPGRADES") localize:NO] autorelease]; @@ -8029,7 +8102,7 @@ void $UIWebDocumentView$_setUIKitDelegate$(UIWebDocumentView *self, SEL sel, id int main(int argc, char *argv[]) { _pooled _trace(); - PackageName = reinterpret_cast(method_getImplementation(class_getInstanceMethod([Package class], @selector(name)))); + PackageName = reinterpret_cast(method_getImplementation(class_getInstanceMethod([Package class], @selector(cyname)))); /* Library Hacks {{{ */ class_addMethod(objc_getClass("DOMNodeList"), @selector(countByEnumeratingWithState:objects:count:), (IMP) &DOMNodeList$countByEnumeratingWithState$objects$count$, "I20@0:4^{NSFastEnumerationState}8^@12I16"); -- 2.45.2