#ifdef LOCAL111 #define _GLIBCXX_DEBUG #else #pragma GCC optimize ("O3") #define NDEBUG #endif // #define _USE_MATH_DEFINES #include const int INF = 1e9; using namespace std; template ostream& operator<< (ostream& os, const pair& p) { os << '(' << p.first << ' ' << p.second << ')'; return os; } #define endl '\n' #define ALL(a) (a).begin(),(a).end() #define SZ(a) int((a).size()) #define FOR(i,a,b) for(int i=(a);i<(b);++i) #define RFOR(i,a,b) for (int i=(b)-1;i>=(a);i--) #define REP(i,n) FOR(i,0,n) #define RREP(i,n) for (int i=(n)-1;i>=0;i--) #ifdef LOCAL111 #define DEBUG(x) cout<<#x<<": "<<(x)< void dpite(T a, T b){ for(T ite = a; ite != b; ite++) cout << (ite == a ? "" : " ") << *ite; cout << endl;} #else #define DEBUG(x) true template void dpite(T a, T b){ return; } #endif #define F first #define S second #define SNP string::npos #define WRC(hoge) cout << "Case #" << (hoge)+1 << ": " template void pite(T a, T b){ for(T ite = a; ite != b; ite++) cout << (ite == a ? "" : " ") << *ite; cout << endl;} template bool chmax(T& a, T b){if(a < b){a = b; return true;} return false;} template bool chmin(T& a, T b){if(a > b){a = b; return true;} return false;} template vector make_v(size_t a){return vector(a);} template auto make_v(size_t a,Ts... ts){ return vector(ts...))>(a,make_v(ts...)); } template typename enable_if::value!=0>::type fill_v(U &u,const V... v){u=U(v...);} template typename enable_if::value==0>::type fill_v(U &u,const V... v){ for(auto &e:u) fill_v(e,v...); } const array dx = {0, 1, 0, -1}; const array dy = {1, 0, -1, 0}; typedef long long int LL; typedef unsigned long long ULL; typedef pair P; void ios_init(){ //cout.setf(ios::fixed); //cout.precision(12); #ifdef LOCAL111 return; #endif ios::sync_with_stdio(false); cin.tie(0); } template class ModInt { public: const static long long mod = MOD; long long x; ModInt() { x = 0; } ModInt(long long x) { x %= mod; this->x = x < 0 ? x+mod : x; } int get() const { return (int)x; } ModInt &operator+=(ModInt that) { if((x += that.get()) >= mod) x -= mod; return *this; } ModInt &operator-=(ModInt that) { if((x += mod-that.get()) >= mod) x -= mod; return *this; } ModInt &operator*=(ModInt that) { x = x*that.get()%mod; return *this; } ModInt &operator/=(ModInt that) { return *this *= that.inverse(); } ModInt operator+(ModInt that) const { return ModInt(*this) += that; } ModInt operator-(ModInt that) const { return ModInt(*this) -= that; } ModInt operator*(ModInt that) const { return ModInt(*this) *= that; } ModInt operator/(ModInt that) const { return ModInt(*this) /= that; } ModInt inverse() const { using std::swap; long long a = x, b = mod, u = 1, v = 0; while(b) { long long t = a/b; a -= t*b; swap(a,b); u -= t*v; swap(u,v); } return ModInt(u); } ModInt pow(int n) const{ ModInt b = *this; ModInt res = 1; while(n != 0) { if(n&1){ res *= b; } b *= b; n >>= 1; } return res; } bool operator==(ModInt that) const { return x == that.get(); } bool operator!=(ModInt that) const { return x != that.get(); } ModInt operator-() const { return x == 0 ? 0 : ModInt(mod-x); } }; template ostream& operator<< (ostream& os, const ModInt& m) { os << m.get(); return os; } template istream& operator>> (istream& is, ModInt& m){ long long n; is >> n; m = n; return is;} typedef ModInt<1000000007> mint; class ConvQuery { vector fac; vector facinv; public: ConvQuery(int n) { fac = vector(n+1); facinv = vector(n+1); fac[0] = 1; facinv[0] = 1; for(int i = 0; i < n; ++i) { fac[i+1] = fac[i]*(i+1); facinv[i+1] = fac[i+1].inverse(); } } mint operator()(int n, int m) { if(n >= 0 and 0 <= m and m <= n) return fac[n]*facinv[n-m]*facinv[m]; else return 0; } }; int main() { ios_init(); LL k, l, r; while(cin >> k >> l >> r) { l--; vector fl, nl, fn, nn; vector fm, nm; fl.push_back(0); nl.push_back(0); fn.push_back(0); nn.push_back(0); fm.push_back(1); nm.push_back(1); auto nlen = [&](LL n) -> int { int res = 0; while(n != 0) { res++; n /= 10; } return res; }; auto dsum = [&](LL n) -> LL { LL res = 0; while(n != 0) { res += n % 10; n /= 10; } return res; }; auto dmul = [&](LL n) -> mint { mint res = 1; while(n != 0) { if(n % 10 == 0) { res *= 10; } else { res *= n % 10; } n /= 10; } return res; }; while(fl.back() < r) { int l = nlen(SZ(fl) * SZ(fl)); nl.push_back(l); fl.push_back(l + 2 * fl.back()); LL ds = dsum(SZ(fn) * SZ(fn)); fn.push_back(ds + 2 * fn.back()); nn.push_back(ds); mint dm = dmul(SZ(fm) * SZ(fm)); fm.push_back(dm * fm.back() * fm.back()); nm.push_back(dm); } if(SZ(fl) - 1 > k) { cout << -1 << endl; continue; } function get_sum = [&](LL x, int i) -> LL { LL m0 = (fl[i] - nl[i]) / 2; LL m1 = m0 + nl[i]; if(m0 <= x && x <= m1) { LL res = 0; int ii = i * i; string sii = to_string(ii); REP(i, x - m0) { res += (sii[i] - '0'); } return res + (fn[i] - nn[i]) / 2; } else if(m1 < x) { return (fn[i] - nn[i]) / 2 + nn[i] + get_sum(x - m1, i - 1); } else { return get_sum(x, i - 1); } }; function get_mul = [&](LL x, int i) -> mint { LL m0 = (fl[i] - nl[i]) / 2; LL m1 = m0 + nl[i]; if(m0 <= x && x <= m1) { mint res = 1; int ii = i * i; string sii = to_string(ii); REP(i, x - m0) { if((sii[i] - '0') == 0) { res *= 10; } else { res *= (sii[i] - '0'); } } DEBUG(res); DEBUG(fm[i - 1]); return res * fm[i - 1]; } else if(m1 < x) { return fm[i - 1] * nm[i] * get_mul(x - m1, i - 1); } else { return get_mul(x, i - 1); } }; // dpite(ALL(fl)); // dpite(ALL(nl)); dpite(ALL(fm)); dpite(ALL(nm)); // DEBUG(get_sum(r, SZ(fl) - 1) ); // DEBUG(get_sum(l, SZ(fl) - 1) ); cout << get_sum(r, SZ(fl) - 1) - get_sum(l, SZ(fl) - 1) << ' ' << get_mul(r, SZ(fl) - 1) / get_mul(l, SZ(fl) - 1) << endl; } return 0; }