00001 
00002 
00003 
00004 
00005 
00006 
00007 
00008 
00009 
00010 
00011 
00012 
00013 
00014 
00015 
00016 
00017 
00018 
00019 
00020 
00021 
00022 
00023 
00024 
00025 
00026 
00027 
00028 
00029 
00030 
00031 
00032 
00033 
00034 
00035 
00036 #ifndef _LOCALE_CLASSES_H
00037 #define _LOCALE_CLASSES_H 1
00038 
00039 #pragma GCC system_header
00040 
00041 #include <bits/localefwd.h>
00042 #include <string>
00043 #include <ext/atomicity.h>
00044 
00045 _GLIBCXX_BEGIN_NAMESPACE(std)
00046 
00047   
00048 
00049 
00050 
00051 
00052 
00053 
00054 
00055 
00056 
00057 
00058 
00059 
00060 
00061 
00062   class locale
00063   {
00064   public:
00065     
00066 
00067     typedef int category;
00068 
00069     
00070     class facet;
00071     class id;
00072     class _Impl;
00073 
00074     friend class facet;
00075     friend class _Impl;
00076 
00077     template<typename _Facet>
00078       friend bool
00079       has_facet(const locale&) throw();
00080 
00081     template<typename _Facet>
00082       friend const _Facet&
00083       use_facet(const locale&);
00084 
00085     template<typename _Cache>
00086       friend struct __use_cache;
00087 
00088 
00089 
00090 
00091 
00092 
00093 
00094 
00095 
00096 
00097 
00098     static const category none      = 0;
00099     static const category ctype     = 1L << 0;
00100     static const category numeric   = 1L << 1;
00101     static const category collate   = 1L << 2;
00102     static const category time      = 1L << 3;
00103     static const category monetary  = 1L << 4;
00104     static const category messages  = 1L << 5;
00105     static const category all       = (ctype | numeric | collate |
00106                        time  | monetary | messages);
00107 
00108 
00109     
00110 
00111 
00112 
00113 
00114 
00115 
00116 
00117     locale() throw();
00118 
00119 
00120 
00121 
00122 
00123 
00124 
00125 
00126     locale(const locale& __other) throw();
00127 
00128 
00129 
00130 
00131 
00132 
00133 
00134 
00135 
00136     explicit
00137     locale(const char* __s);
00138 
00139 
00140 
00141 
00142 
00143 
00144 
00145 
00146 
00147 
00148 
00149 
00150 
00151     locale(const locale& __base, const char* __s, category __cat);
00152 
00153 
00154 
00155 
00156 
00157 
00158 
00159 
00160 
00161 
00162 
00163 
00164     locale(const locale& __base, const locale& __add, category __cat);
00165 
00166 
00167 
00168 
00169 
00170 
00171 
00172 
00173 
00174 
00175 
00176     template<typename _Facet>
00177       locale(const locale& __other, _Facet* __f);
00178 
00179 
00180     ~locale() throw();
00181 
00182 
00183 
00184 
00185 
00186 
00187 
00188 
00189 
00190     const locale&
00191     operator=(const locale& __other) throw();
00192 
00193 
00194 
00195 
00196 
00197 
00198 
00199 
00200 
00201 
00202 
00203 
00204 
00205     template<typename _Facet>
00206       locale
00207       combine(const locale& __other) const;
00208 
00209     
00210 
00211 
00212 
00213 
00214     string
00215     name() const;
00216 
00217 
00218 
00219 
00220 
00221 
00222 
00223 
00224     bool
00225     operator==(const locale& __other) const throw();
00226 
00227 
00228 
00229 
00230 
00231 
00232 
00233     bool
00234     operator!=(const locale& __other) const throw()
00235     { return !(this->operator==(__other)); }
00236 
00237 
00238 
00239 
00240 
00241 
00242 
00243 
00244 
00245 
00246 
00247 
00248 
00249 
00250 
00251 
00252     template<typename _Char, typename _Traits, typename _Alloc>
00253       bool
00254       operator()(const basic_string<_Char, _Traits, _Alloc>& __s1,
00255          const basic_string<_Char, _Traits, _Alloc>& __s2) const;
00256 
00257     
00258 
00259 
00260 
00261 
00262 
00263 
00264 
00265 
00266 
00267 
00268     static locale
00269     global(const locale&);
00270 
00271 
00272 
00273 
00274     static const locale&
00275     classic();
00276 
00277   private:
00278     
00279     _Impl*      _M_impl;
00280 
00281     
00282     static _Impl*       _S_classic;
00283 
00284     
00285     static _Impl*   _S_global;
00286 
00287     
00288     
00289     
00290     
00291     static const char* const* const _S_categories;
00292 
00293     
00294     
00295     
00296     
00297     
00298     
00299     
00300     
00301     
00302     
00303     enum { _S_categories_size = 6 + _GLIBCXX_NUM_CATEGORIES };
00304 
00305 #ifdef __GTHREADS
00306     static __gthread_once_t _S_once;
00307 #endif
00308 
00309     explicit
00310     locale(_Impl*) throw();
00311 
00312     static void
00313     _S_initialize();
00314 
00315     static void
00316     _S_initialize_once() throw();
00317 
00318     static category
00319     _S_normalize_category(category);
00320 
00321     void
00322     _M_coalesce(const locale& __base, const locale& __add, category __cat);
00323   };
00324 
00325 
00326   
00327 
00328 
00329 
00330 
00331 
00332 
00333 
00334 
00335 
00336 
00337   class locale::facet
00338   {
00339   private:
00340     friend class locale;
00341     friend class locale::_Impl;
00342 
00343     mutable _Atomic_word        _M_refcount;
00344 
00345     
00346     static __c_locale                   _S_c_locale;
00347 
00348     
00349     static const char           _S_c_name[2];
00350 
00351 #ifdef __GTHREADS
00352     static __gthread_once_t     _S_once;
00353 #endif
00354 
00355     static void
00356     _S_initialize_once();
00357 
00358   protected:
00359 
00360 
00361 
00362 
00363 
00364 
00365 
00366 
00367 
00368     explicit
00369     facet(size_t __refs = 0) throw() : _M_refcount(__refs ? 1 : 0)
00370     { }
00371 
00372 
00373     virtual
00374     ~facet();
00375 
00376     static void
00377     _S_create_c_locale(__c_locale& __cloc, const char* __s,
00378                __c_locale __old = 0);
00379 
00380     static __c_locale
00381     _S_clone_c_locale(__c_locale& __cloc) throw();
00382 
00383     static void
00384     _S_destroy_c_locale(__c_locale& __cloc);
00385 
00386     static __c_locale
00387     _S_lc_ctype_c_locale(__c_locale __cloc, const char* __s);
00388 
00389     
00390     
00391     static __c_locale
00392     _S_get_c_locale();
00393 
00394     _GLIBCXX_CONST static const char*
00395     _S_get_c_name() throw();
00396 
00397   private:
00398     void
00399     _M_add_reference() const throw()
00400     { __gnu_cxx::__atomic_add_dispatch(&_M_refcount, 1); }
00401 
00402     void
00403     _M_remove_reference() const throw()
00404     {
00405       if (__gnu_cxx::__exchange_and_add_dispatch(&_M_refcount, -1) == 1)
00406     {
00407       __try
00408         { delete this; }
00409       __catch(...)
00410         { }
00411     }
00412     }
00413 
00414     facet(const facet&);  
00415 
00416     facet&
00417     operator=(const facet&);  
00418   };
00419 
00420 
00421   
00422 
00423 
00424 
00425 
00426 
00427 
00428 
00429 
00430 
00431 
00432   class locale::id
00433   {
00434   private:
00435     friend class locale;
00436     friend class locale::_Impl;
00437 
00438     template<typename _Facet>
00439       friend const _Facet&
00440       use_facet(const locale&);
00441 
00442     template<typename _Facet>
00443       friend bool
00444       has_facet(const locale&) throw();
00445 
00446     
00447     
00448     
00449     mutable size_t      _M_index;
00450 
00451     
00452     static _Atomic_word     _S_refcount;
00453 
00454     void
00455     operator=(const id&);  
00456 
00457     id(const id&);  
00458 
00459   public:
00460     
00461     
00462 
00463     id() { }
00464 
00465     size_t
00466     _M_id() const throw();
00467   };
00468 
00469 
00470   
00471   class locale::_Impl
00472   {
00473   public:
00474     
00475     friend class locale;
00476     friend class locale::facet;
00477 
00478     template<typename _Facet>
00479       friend bool
00480       has_facet(const locale&) throw();
00481 
00482     template<typename _Facet>
00483       friend const _Facet&
00484       use_facet(const locale&);
00485 
00486     template<typename _Cache>
00487       friend struct __use_cache;
00488 
00489   private:
00490     
00491     _Atomic_word            _M_refcount;
00492     const facet**           _M_facets;
00493     size_t              _M_facets_size;
00494     const facet**           _M_caches;
00495     char**              _M_names;
00496     static const locale::id* const  _S_id_ctype[];
00497     static const locale::id* const  _S_id_numeric[];
00498     static const locale::id* const  _S_id_collate[];
00499     static const locale::id* const  _S_id_time[];
00500     static const locale::id* const  _S_id_monetary[];
00501     static const locale::id* const  _S_id_messages[];
00502     static const locale::id* const* const _S_facet_categories[];
00503 
00504     void
00505     _M_add_reference() throw()
00506     { __gnu_cxx::__atomic_add_dispatch(&_M_refcount, 1); }
00507 
00508     void
00509     _M_remove_reference() throw()
00510     {
00511       if (__gnu_cxx::__exchange_and_add_dispatch(&_M_refcount, -1) == 1)
00512     {
00513       __try
00514         { delete this; }
00515       __catch(...)
00516         { }
00517     }
00518     }
00519 
00520     _Impl(const _Impl&, size_t);
00521     _Impl(const char*, size_t);
00522     _Impl(size_t) throw();
00523 
00524    ~_Impl() throw();
00525 
00526     _Impl(const _Impl&);  
00527 
00528     void
00529     operator=(const _Impl&);  
00530 
00531     bool
00532     _M_check_same_name()
00533     {
00534       bool __ret = true;
00535       if (_M_names[1])
00536     
00537     for (size_t __i = 0; __ret && __i < _S_categories_size - 1; ++__i)
00538       __ret = __builtin_strcmp(_M_names[__i], _M_names[__i + 1]) == 0;
00539       return __ret;
00540     }
00541 
00542     void
00543     _M_replace_categories(const _Impl*, category);
00544 
00545     void
00546     _M_replace_category(const _Impl*, const locale::id* const*);
00547 
00548     void
00549     _M_replace_facet(const _Impl*, const locale::id*);
00550 
00551     void
00552     _M_install_facet(const locale::id*, const facet*);
00553 
00554     template<typename _Facet>
00555       void
00556       _M_init_facet(_Facet* __facet)
00557       { _M_install_facet(&_Facet::id, __facet); }
00558 
00559     void
00560     _M_install_cache(const facet*, size_t);
00561   };
00562 
00563 
00564 
00565 
00566 
00567 
00568 
00569 
00570 
00571 
00572 
00573 
00574 
00575   template<typename _Facet>
00576     bool
00577     has_facet(const locale& __loc) throw();
00578 
00579 
00580 
00581 
00582 
00583 
00584 
00585 
00586 
00587 
00588 
00589 
00590 
00591 
00592   template<typename _Facet>
00593     const _Facet&
00594     use_facet(const locale& __loc);
00595 
00596 
00597 
00598 
00599 
00600 
00601 
00602 
00603 
00604 
00605 
00606 
00607 
00608 
00609   template<typename _CharT>
00610     class collate : public locale::facet
00611     {
00612     public:
00613       
00614 
00615 
00616       typedef _CharT            char_type;
00617       typedef basic_string<_CharT>  string_type;
00618 
00619 
00620     protected:
00621       
00622       
00623       __c_locale            _M_c_locale_collate;
00624 
00625     public:
00626 
00627       static locale::id         id;
00628 
00629 
00630 
00631 
00632 
00633 
00634 
00635 
00636       explicit
00637       collate(size_t __refs = 0)
00638       : facet(__refs), _M_c_locale_collate(_S_get_c_locale())
00639       { }
00640 
00641 
00642 
00643 
00644 
00645 
00646 
00647 
00648 
00649 
00650       explicit
00651       collate(__c_locale __cloc, size_t __refs = 0)
00652       : facet(__refs), _M_c_locale_collate(_S_clone_c_locale(__cloc))
00653       { }
00654 
00655 
00656 
00657 
00658 
00659 
00660 
00661 
00662 
00663 
00664 
00665 
00666 
00667       int
00668       compare(const _CharT* __lo1, const _CharT* __hi1,
00669           const _CharT* __lo2, const _CharT* __hi2) const
00670       { return this->do_compare(__lo1, __hi1, __lo2, __hi2); }
00671 
00672 
00673 
00674 
00675 
00676 
00677 
00678 
00679 
00680 
00681 
00682 
00683 
00684 
00685 
00686       string_type
00687       transform(const _CharT* __lo, const _CharT* __hi) const
00688       { return this->do_transform(__lo, __hi); }
00689 
00690 
00691 
00692 
00693 
00694 
00695 
00696 
00697 
00698 
00699 
00700       long
00701       hash(const _CharT* __lo, const _CharT* __hi) const
00702       { return this->do_hash(__lo, __hi); }
00703 
00704       
00705       int
00706       _M_compare(const _CharT*, const _CharT*) const throw();
00707 
00708       size_t
00709       _M_transform(_CharT*, const _CharT*, size_t) const throw();
00710 
00711   protected:
00712 
00713       virtual
00714       ~collate()
00715       { _S_destroy_c_locale(_M_c_locale_collate); }
00716 
00717 
00718 
00719 
00720 
00721 
00722 
00723 
00724 
00725 
00726 
00727 
00728 
00729       virtual int
00730       do_compare(const _CharT* __lo1, const _CharT* __hi1,
00731          const _CharT* __lo2, const _CharT* __hi2) const;
00732 
00733 
00734 
00735 
00736 
00737 
00738 
00739 
00740 
00741 
00742 
00743 
00744 
00745       virtual string_type
00746       do_transform(const _CharT* __lo, const _CharT* __hi) const;
00747 
00748 
00749 
00750 
00751 
00752 
00753 
00754 
00755 
00756 
00757 
00758       virtual long
00759       do_hash(const _CharT* __lo, const _CharT* __hi) const;
00760     };
00761 
00762   template<typename _CharT>
00763     locale::id collate<_CharT>::id;
00764 
00765   
00766   template<>
00767     int
00768     collate<char>::_M_compare(const char*, const char*) const throw();
00769 
00770   template<>
00771     size_t
00772     collate<char>::_M_transform(char*, const char*, size_t) const throw();
00773 
00774 #ifdef _GLIBCXX_USE_WCHAR_T
00775   template<>
00776     int
00777     collate<wchar_t>::_M_compare(const wchar_t*, const wchar_t*) const throw();
00778 
00779   template<>
00780     size_t
00781     collate<wchar_t>::_M_transform(wchar_t*, const wchar_t*, size_t) const throw();
00782 #endif
00783 
00784 
00785   template<typename _CharT>
00786     class collate_byname : public collate<_CharT>
00787     {
00788     public:
00789 
00790 
00791       typedef _CharT               char_type;
00792       typedef basic_string<_CharT> string_type;
00793 
00794 
00795       explicit
00796       collate_byname(const char* __s, size_t __refs = 0)
00797       : collate<_CharT>(__refs)
00798       {
00799     if (__builtin_strcmp(__s, "C") != 0
00800         && __builtin_strcmp(__s, "POSIX") != 0)
00801       {
00802         this->_S_destroy_c_locale(this->_M_c_locale_collate);
00803         this->_S_create_c_locale(this->_M_c_locale_collate, __s);
00804       }
00805       }
00806 
00807     protected:
00808       virtual
00809       ~collate_byname() { }
00810     };
00811 
00812 _GLIBCXX_END_NAMESPACE
00813 
00814 #ifndef _GLIBCXX_EXPORT_TEMPLATE
00815 # include <bits/locale_classes.tcc>
00816 #endif
00817 
00818 #endif