|
|
@ -1,526 +0,0 @@ |
|
|
|
-module(rumFormat). |
|
|
|
|
|
|
|
%% fork of io_lib_format that uses trunc_io to protect against large terms |
|
|
|
|
|
|
|
-export([format/3, format/4]). |
|
|
|
|
|
|
|
-record(options, { |
|
|
|
chomp = false :: boolean() |
|
|
|
}). |
|
|
|
|
|
|
|
format(FmtStr, Args, MaxLen) -> |
|
|
|
format(FmtStr, Args, MaxLen, []). |
|
|
|
|
|
|
|
format([], [], _, _) -> |
|
|
|
""; |
|
|
|
format(FmtStr, Args, MaxLen, Opts) when is_atom(FmtStr) -> |
|
|
|
format(atom_to_list(FmtStr), Args, MaxLen, Opts); |
|
|
|
format(FmtStr, Args, MaxLen, Opts) when is_binary(FmtStr) -> |
|
|
|
format(binary_to_list(FmtStr), Args, MaxLen, Opts); |
|
|
|
format(FmtStr, Args, MaxLen, Opts) when is_list(FmtStr) -> |
|
|
|
case io_lib:deep_char_list(FmtStr) of |
|
|
|
true -> |
|
|
|
Options = make_options(Opts, #options{}), |
|
|
|
Cs = collect(FmtStr, Args), |
|
|
|
{Cs2, MaxLen2} = build(Cs, [], MaxLen, Options), |
|
|
|
%% count how many terms remain |
|
|
|
{Count, StrLen} = lists:foldl( |
|
|
|
fun({_C, _As, _F, _Adj, _P, _Pad, _Enc}, {Terms, Chars}) -> |
|
|
|
{Terms + 1, Chars}; |
|
|
|
(_, {Terms, Chars}) -> |
|
|
|
{Terms, Chars + 1} |
|
|
|
end, {0, 0}, Cs2), |
|
|
|
build2(Cs2, Count, MaxLen2 - StrLen); |
|
|
|
false -> |
|
|
|
erlang:error(badarg) |
|
|
|
end; |
|
|
|
format(_FmtStr, _Args, _MaxLen, _Opts) -> |
|
|
|
erlang:error(badarg). |
|
|
|
|
|
|
|
collect([$~ | Fmt0], Args0) -> |
|
|
|
{C, Fmt1, Args1} = collect_cseq(Fmt0, Args0), |
|
|
|
[C | collect(Fmt1, Args1)]; |
|
|
|
collect([C | Fmt], Args) -> |
|
|
|
[C | collect(Fmt, Args)]; |
|
|
|
collect([], []) -> []. |
|
|
|
|
|
|
|
collect_cseq(Fmt0, Args0) -> |
|
|
|
{F, Ad, Fmt1, Args1} = field_width(Fmt0, Args0), |
|
|
|
{P, Fmt2, Args2} = precision(Fmt1, Args1), |
|
|
|
{Pad, Fmt3, Args3} = pad_char(Fmt2, Args2), |
|
|
|
{Encoding, Fmt4, Args4} = encoding(Fmt3, Args3), |
|
|
|
{C, As, Fmt5, Args5} = collect_cc(Fmt4, Args4), |
|
|
|
{{C, As, F, Ad, P, Pad, Encoding}, Fmt5, Args5}. |
|
|
|
|
|
|
|
encoding([$t | Fmt], Args) -> |
|
|
|
{unicode, Fmt, Args}; |
|
|
|
encoding(Fmt, Args) -> |
|
|
|
{latin1, Fmt, Args}. |
|
|
|
|
|
|
|
field_width([$- | Fmt0], Args0) -> |
|
|
|
{F, Fmt, Args} = field_value(Fmt0, Args0), |
|
|
|
field_width(-F, Fmt, Args); |
|
|
|
field_width(Fmt0, Args0) -> |
|
|
|
{F, Fmt, Args} = field_value(Fmt0, Args0), |
|
|
|
field_width(F, Fmt, Args). |
|
|
|
|
|
|
|
field_width(F, Fmt, Args) when F < 0 -> |
|
|
|
{-F, left, Fmt, Args}; |
|
|
|
field_width(F, Fmt, Args) when F >= 0 -> |
|
|
|
{F, right, Fmt, Args}. |
|
|
|
|
|
|
|
precision([$. | Fmt], Args) -> |
|
|
|
field_value(Fmt, Args); |
|
|
|
precision(Fmt, Args) -> |
|
|
|
{none, Fmt, Args}. |
|
|
|
|
|
|
|
field_value([$* | Fmt], [A | Args]) when is_integer(A) -> |
|
|
|
{A, Fmt, Args}; |
|
|
|
field_value([C | Fmt], Args) when is_integer(C), C >= $0, C =< $9 -> |
|
|
|
field_value([C | Fmt], Args, 0); |
|
|
|
field_value(Fmt, Args) -> |
|
|
|
{none, Fmt, Args}. |
|
|
|
|
|
|
|
field_value([C | Fmt], Args, F) when is_integer(C), C >= $0, C =< $9 -> |
|
|
|
field_value(Fmt, Args, 10 * F + (C - $0)); |
|
|
|
field_value(Fmt, Args, F) -> %Default case |
|
|
|
{F, Fmt, Args}. |
|
|
|
|
|
|
|
pad_char([$., $* | Fmt], [Pad | Args]) -> {Pad, Fmt, Args}; |
|
|
|
pad_char([$., Pad | Fmt], Args) -> {Pad, Fmt, Args}; |
|
|
|
pad_char(Fmt, Args) -> {$\s, Fmt, Args}. |
|
|
|
|
|
|
|
%% collect_cc([FormatChar], [Argument]) -> |
|
|
|
%% {Control,[ControlArg],[FormatChar],[Arg]}. |
|
|
|
%% Here we collect the argments for each control character. |
|
|
|
%% Be explicit to cause failure early. |
|
|
|
|
|
|
|
collect_cc([$w | Fmt], [A | Args]) -> {$w, [A], Fmt, Args}; |
|
|
|
collect_cc([$p | Fmt], [A | Args]) -> {$p, [A], Fmt, Args}; |
|
|
|
collect_cc([$W | Fmt], [A, Depth | Args]) -> {$W, [A, Depth], Fmt, Args}; |
|
|
|
collect_cc([$P | Fmt], [A, Depth | Args]) -> {$P, [A, Depth], Fmt, Args}; |
|
|
|
collect_cc([$s | Fmt], [A | Args]) -> {$s, [A], Fmt, Args}; |
|
|
|
collect_cc([$e | Fmt], [A | Args]) -> {$e, [A], Fmt, Args}; |
|
|
|
collect_cc([$f | Fmt], [A | Args]) -> {$f, [A], Fmt, Args}; |
|
|
|
collect_cc([$g | Fmt], [A | Args]) -> {$g, [A], Fmt, Args}; |
|
|
|
collect_cc([$b | Fmt], [A | Args]) -> {$b, [A], Fmt, Args}; |
|
|
|
collect_cc([$B | Fmt], [A | Args]) -> {$B, [A], Fmt, Args}; |
|
|
|
collect_cc([$x | Fmt], [A, Prefix | Args]) -> {$x, [A, Prefix], Fmt, Args}; |
|
|
|
collect_cc([$X | Fmt], [A, Prefix | Args]) -> {$X, [A, Prefix], Fmt, Args}; |
|
|
|
collect_cc([$+ | Fmt], [A | Args]) -> {$+, [A], Fmt, Args}; |
|
|
|
collect_cc([$# | Fmt], [A | Args]) -> {$#, [A], Fmt, Args}; |
|
|
|
collect_cc([$c | Fmt], [A | Args]) -> {$c, [A], Fmt, Args}; |
|
|
|
collect_cc([$~ | Fmt], Args) when is_list(Args) -> {$~, [], Fmt, Args}; |
|
|
|
collect_cc([$n | Fmt], Args) when is_list(Args) -> {$n, [], Fmt, Args}; |
|
|
|
collect_cc([$i | Fmt], [A | Args]) -> {$i, [A], Fmt, Args}. |
|
|
|
|
|
|
|
|
|
|
|
%% build([Control], Pc, Indentation) -> [Char]. |
|
|
|
%% Interpret the control structures. Count the number of print |
|
|
|
%% remaining and only calculate indentation when necessary. Must also |
|
|
|
%% be smart when calculating indentation for characters in format. |
|
|
|
|
|
|
|
build([{$n, _, _, _, _, _, _}], Acc, MaxLen, #options{chomp = true}) -> |
|
|
|
%% trailing ~n, ignore |
|
|
|
{lists:reverse(Acc), MaxLen}; |
|
|
|
build([{C, As, F, Ad, P, Pad, Enc} | Cs], Acc, MaxLen, O) -> |
|
|
|
{S, MaxLen2} = control(C, As, F, Ad, P, Pad, Enc, MaxLen), |
|
|
|
build(Cs, [S | Acc], MaxLen2, O); |
|
|
|
build([$\n], Acc, MaxLen, #options{chomp = true}) -> |
|
|
|
%% trailing \n, ignore |
|
|
|
{lists:reverse(Acc), MaxLen}; |
|
|
|
build([$\n | Cs], Acc, MaxLen, O) -> |
|
|
|
build(Cs, [$\n | Acc], MaxLen - 1, O); |
|
|
|
build([$\t | Cs], Acc, MaxLen, O) -> |
|
|
|
build(Cs, [$\t | Acc], MaxLen - 1, O); |
|
|
|
build([C | Cs], Acc, MaxLen, O) -> |
|
|
|
build(Cs, [C | Acc], MaxLen - 1, O); |
|
|
|
build([], Acc, MaxLen, _O) -> |
|
|
|
{lists:reverse(Acc), MaxLen}. |
|
|
|
|
|
|
|
build2([{C, As, F, Ad, P, Pad, Enc} | Cs], Count, MaxLen) -> |
|
|
|
{S, Len} = control2(C, As, F, Ad, P, Pad, Enc, MaxLen div Count), |
|
|
|
[S | build2(Cs, Count - 1, MaxLen - Len)]; |
|
|
|
build2([C | Cs], Count, MaxLen) -> |
|
|
|
[C | build2(Cs, Count, MaxLen)]; |
|
|
|
build2([], _, _) -> []. |
|
|
|
|
|
|
|
%% control(FormatChar, [Argument], FieldWidth, Adjust, Precision, PadChar, |
|
|
|
%% Indentation) -> [Char] |
|
|
|
%% This is the main dispatch function for the various formatting commands. |
|
|
|
%% Field widths and precisions have already been calculated. |
|
|
|
|
|
|
|
control($e, [A], F, Adj, P, Pad, _Enc, L) when is_float(A) -> |
|
|
|
Res = fwrite_e(A, F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($f, [A], F, Adj, P, Pad, _Enc, L) when is_float(A) -> |
|
|
|
Res = fwrite_f(A, F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($g, [A], F, Adj, P, Pad, _Enc, L) when is_float(A) -> |
|
|
|
Res = fwrite_g(A, F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($b, [A], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
Res = unprefixed_integer(A, F, Adj, base(P), Pad, true), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($B, [A], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
Res = unprefixed_integer(A, F, Adj, base(P), Pad, false), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($x, [A, Prefix], F, Adj, P, Pad, _Enc, L) when is_integer(A), |
|
|
|
is_atom(Prefix) -> |
|
|
|
Res = prefixed_integer(A, F, Adj, base(P), Pad, atom_to_list(Prefix), true), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($x, [A, Prefix], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
true = io_lib:deep_char_list(Prefix), %Check if Prefix a character list |
|
|
|
Res = prefixed_integer(A, F, Adj, base(P), Pad, Prefix, true), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($X, [A, Prefix], F, Adj, P, Pad, _Enc, L) when is_integer(A), |
|
|
|
is_atom(Prefix) -> |
|
|
|
Res = prefixed_integer(A, F, Adj, base(P), Pad, atom_to_list(Prefix), false), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($X, [A, Prefix], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
true = io_lib:deep_char_list(Prefix), %Check if Prefix a character list |
|
|
|
Res = prefixed_integer(A, F, Adj, base(P), Pad, Prefix, false), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($+, [A], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
Base = base(P), |
|
|
|
Prefix = [integer_to_list(Base), $#], |
|
|
|
Res = prefixed_integer(A, F, Adj, Base, Pad, Prefix, true), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($#, [A], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
Base = base(P), |
|
|
|
Prefix = [integer_to_list(Base), $#], |
|
|
|
Res = prefixed_integer(A, F, Adj, Base, Pad, Prefix, false), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($c, [A], F, Adj, P, Pad, unicode, L) when is_integer(A) -> |
|
|
|
Res = char(A, F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($c, [A], F, Adj, P, Pad, _Enc, L) when is_integer(A) -> |
|
|
|
Res = char(A band 255, F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($~, [], F, Adj, P, Pad, _Enc, L) -> |
|
|
|
Res = char($~, F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($n, [], F, Adj, P, Pad, _Enc, L) -> |
|
|
|
Res = newline(F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control($i, [_A], _F, _Adj, _P, _Pad, _Enc, L) -> |
|
|
|
{[], L}; |
|
|
|
control($s, [A], F, Adj, P, Pad, _Enc, L) when is_atom(A) -> |
|
|
|
Res = string(atom_to_list(A), F, Adj, P, Pad), |
|
|
|
{Res, L - lists:flatlength(Res)}; |
|
|
|
control(C, A, F, Adj, P, Pad, Enc, L) -> |
|
|
|
%% save this for later - these are all the 'large' terms |
|
|
|
{{C, A, F, Adj, P, Pad, Enc}, L}. |
|
|
|
|
|
|
|
control2($w, [A], F, Adj, P, Pad, _Enc, L) -> |
|
|
|
Term = rumTruncIo:fprint(A, L, [{lists_as_strings, false}]), |
|
|
|
Res = term(Term, F, Adj, P, Pad), |
|
|
|
{Res, lists:flatlength(Res)}; |
|
|
|
control2($p, [A], _F, _Adj, _P, _Pad, _Enc, L) -> |
|
|
|
Term = rumTruncIo:fprint(A, L, [{lists_as_strings, true}]), |
|
|
|
{Term, lists:flatlength(Term)}; |
|
|
|
control2($W, [A, Depth], F, Adj, P, Pad, _Enc, L) when is_integer(Depth) -> |
|
|
|
Term = rumTruncIo:fprint(A, L, [{depth, Depth}, {lists_as_strings, false}]), |
|
|
|
Res = term(Term, F, Adj, P, Pad), |
|
|
|
{Res, lists:flatlength(Res)}; |
|
|
|
control2($P, [A, Depth], _F, _Adj, _P, _Pad, _Enc, L) when is_integer(Depth) -> |
|
|
|
Term = rumTruncIo:fprint(A, L, [{depth, Depth}, {lists_as_strings, true}]), |
|
|
|
{Term, lists:flatlength(Term)}; |
|
|
|
control2($s, [L0], F, Adj, P, Pad, latin1, L) -> |
|
|
|
List = rumTruncIo:fprint(iolist_to_chars(L0), L, [{force_strings, true}]), |
|
|
|
Res = string(List, F, Adj, P, Pad), |
|
|
|
{Res, lists:flatlength(Res)}; |
|
|
|
control2($s, [L0], F, Adj, P, Pad, unicode, L) -> |
|
|
|
List = rumTruncIo:fprint(cdata_to_chars(L0), L, [{force_strings, true}]), |
|
|
|
Res = uniconv(string(List, F, Adj, P, Pad)), |
|
|
|
{Res, lists:flatlength(Res)}. |
|
|
|
|
|
|
|
iolist_to_chars([C | Cs]) when is_integer(C), C >= $\000, C =< $\377 -> |
|
|
|
[C | iolist_to_chars(Cs)]; |
|
|
|
iolist_to_chars([I | Cs]) -> |
|
|
|
[iolist_to_chars(I) | iolist_to_chars(Cs)]; |
|
|
|
iolist_to_chars([]) -> |
|
|
|
[]; |
|
|
|
iolist_to_chars(B) when is_binary(B) -> |
|
|
|
binary_to_list(B). |
|
|
|
|
|
|
|
cdata_to_chars([C | Cs]) when is_integer(C), C >= $\000 -> |
|
|
|
[C | cdata_to_chars(Cs)]; |
|
|
|
cdata_to_chars([I | Cs]) -> |
|
|
|
[cdata_to_chars(I) | cdata_to_chars(Cs)]; |
|
|
|
cdata_to_chars([]) -> |
|
|
|
[]; |
|
|
|
cdata_to_chars(B) when is_binary(B) -> |
|
|
|
case catch unicode:characters_to_list(B) of |
|
|
|
L when is_list(L) -> L; |
|
|
|
_ -> binary_to_list(B) |
|
|
|
end. |
|
|
|
|
|
|
|
make_options([], Options) -> |
|
|
|
Options; |
|
|
|
make_options([{chomp, Bool} | T], Options) when is_boolean(Bool) -> |
|
|
|
make_options(T, Options#options{chomp = Bool}). |
|
|
|
|
|
|
|
-ifdef(UNICODE_AS_BINARIES). |
|
|
|
uniconv(C) -> |
|
|
|
unicode:characters_to_binary(C, unicode). |
|
|
|
-else. |
|
|
|
uniconv(C) -> |
|
|
|
C. |
|
|
|
-endif. |
|
|
|
%% Default integer base |
|
|
|
base(none) -> |
|
|
|
10; |
|
|
|
base(B) when is_integer(B) -> |
|
|
|
B. |
|
|
|
|
|
|
|
%% term(TermList, Field, Adjust, Precision, PadChar) |
|
|
|
%% Output the characters in a term. |
|
|
|
%% Adjust the characters within the field if length less than Max padding |
|
|
|
%% with PadChar. |
|
|
|
|
|
|
|
term(T, none, _Adj, none, _Pad) -> T; |
|
|
|
term(T, none, Adj, P, Pad) -> term(T, P, Adj, P, Pad); |
|
|
|
term(T, F, Adj, P0, Pad) -> |
|
|
|
L = lists:flatlength(T), |
|
|
|
P = case P0 of none -> erlang:min(L, F); _ -> P0 end, |
|
|
|
if |
|
|
|
L > P -> |
|
|
|
adjust(chars($*, P), chars(Pad, F - P), Adj); |
|
|
|
F >= P -> |
|
|
|
adjust(T, chars(Pad, F - L), Adj) |
|
|
|
end. |
|
|
|
|
|
|
|
%% fwrite_e(Float, Field, Adjust, Precision, PadChar) |
|
|
|
|
|
|
|
fwrite_e(Fl, none, Adj, none, Pad) -> %Default values |
|
|
|
fwrite_e(Fl, none, Adj, 6, Pad); |
|
|
|
fwrite_e(Fl, none, _Adj, P, _Pad) when P >= 2 -> |
|
|
|
float_e(Fl, float_data(Fl), P); |
|
|
|
fwrite_e(Fl, F, Adj, none, Pad) -> |
|
|
|
fwrite_e(Fl, F, Adj, 6, Pad); |
|
|
|
fwrite_e(Fl, F, Adj, P, Pad) when P >= 2 -> |
|
|
|
term(float_e(Fl, float_data(Fl), P), F, Adj, F, Pad). |
|
|
|
|
|
|
|
float_e(Fl, Fd, P) when Fl < 0.0 -> %Negative numbers |
|
|
|
[$- | float_e(-Fl, Fd, P)]; |
|
|
|
float_e(_Fl, {Ds, E}, P) -> |
|
|
|
case float_man(Ds, 1, P - 1) of |
|
|
|
{[$0 | Fs], true} -> [[$1 | Fs] | float_exp(E)]; |
|
|
|
{Fs, false} -> [Fs | float_exp(E - 1)] |
|
|
|
end. |
|
|
|
|
|
|
|
%% float_man([Digit], Icount, Dcount) -> {[Chars],CarryFlag}. |
|
|
|
%% Generate the characters in the mantissa from the digits with Icount |
|
|
|
%% characters before the '.' and Dcount decimals. Handle carry and let |
|
|
|
%% caller decide what to do at top. |
|
|
|
|
|
|
|
float_man(Ds, 0, Dc) -> |
|
|
|
{Cs, C} = float_man(Ds, Dc), |
|
|
|
{[$. | Cs], C}; |
|
|
|
float_man([D | Ds], I, Dc) -> |
|
|
|
case float_man(Ds, I - 1, Dc) of |
|
|
|
{Cs, true} when D =:= $9 -> {[$0 | Cs], true}; |
|
|
|
{Cs, true} -> {[D + 1 | Cs], false}; |
|
|
|
{Cs, false} -> {[D | Cs], false} |
|
|
|
end; |
|
|
|
float_man([], I, Dc) -> %Pad with 0's |
|
|
|
{string:chars($0, I, [$. | string:chars($0, Dc)]), false}. |
|
|
|
|
|
|
|
float_man([D | _], 0) when D >= $5 -> {[], true}; |
|
|
|
float_man([_ | _], 0) -> {[], false}; |
|
|
|
float_man([D | Ds], Dc) -> |
|
|
|
case float_man(Ds, Dc - 1) of |
|
|
|
{Cs, true} when D =:= $9 -> {[$0 | Cs], true}; |
|
|
|
{Cs, true} -> {[D + 1 | Cs], false}; |
|
|
|
{Cs, false} -> {[D | Cs], false} |
|
|
|
end; |
|
|
|
float_man([], Dc) -> {string:chars($0, Dc), false}. %Pad with 0's |
|
|
|
|
|
|
|
%% float_exp(Exponent) -> [Char]. |
|
|
|
%% Generate the exponent of a floating point number. Always include sign. |
|
|
|
|
|
|
|
float_exp(E) when E >= 0 -> |
|
|
|
[$e, $+ | integer_to_list(E)]; |
|
|
|
float_exp(E) -> |
|
|
|
[$e | integer_to_list(E)]. |
|
|
|
|
|
|
|
%% fwrite_f(FloatData, Field, Adjust, Precision, PadChar) |
|
|
|
|
|
|
|
fwrite_f(Fl, none, Adj, none, Pad) -> %Default values |
|
|
|
fwrite_f(Fl, none, Adj, 6, Pad); |
|
|
|
fwrite_f(Fl, none, _Adj, P, _Pad) when P >= 1 -> |
|
|
|
float_f(Fl, float_data(Fl), P); |
|
|
|
fwrite_f(Fl, F, Adj, none, Pad) -> |
|
|
|
fwrite_f(Fl, F, Adj, 6, Pad); |
|
|
|
fwrite_f(Fl, F, Adj, P, Pad) when P >= 1 -> |
|
|
|
term(float_f(Fl, float_data(Fl), P), F, Adj, F, Pad). |
|
|
|
|
|
|
|
float_f(Fl, Fd, P) when Fl < 0.0 -> |
|
|
|
[$- | float_f(-Fl, Fd, P)]; |
|
|
|
float_f(Fl, {Ds, E}, P) when E =< 0 -> |
|
|
|
float_f(Fl, {string:chars($0, -E + 1, Ds), 1}, P); %Prepend enough 0's |
|
|
|
float_f(_Fl, {Ds, E}, P) -> |
|
|
|
case float_man(Ds, E, P) of |
|
|
|
{Fs, true} -> "1" ++ Fs; %Handle carry |
|
|
|
{Fs, false} -> Fs |
|
|
|
end. |
|
|
|
|
|
|
|
%% float_data([FloatChar]) -> {[Digit],Exponent} |
|
|
|
|
|
|
|
float_data(Fl) -> |
|
|
|
float_data(float_to_list(Fl), []). |
|
|
|
|
|
|
|
float_data([$e | E], Ds) -> |
|
|
|
{lists:reverse(Ds), list_to_integer(E) + 1}; |
|
|
|
float_data([D | Cs], Ds) when D >= $0, D =< $9 -> |
|
|
|
float_data(Cs, [D | Ds]); |
|
|
|
float_data([_ | Cs], Ds) -> |
|
|
|
float_data(Cs, Ds). |
|
|
|
|
|
|
|
%% fwrite_g(Float, Field, Adjust, Precision, PadChar) |
|
|
|
%% Use the f form if Float is >= 0.1 and < 1.0e4, |
|
|
|
%% and the prints correctly in the f form, else the e form. |
|
|
|
%% Precision always means the # of significant digits. |
|
|
|
|
|
|
|
fwrite_g(Fl, F, Adj, none, Pad) -> |
|
|
|
fwrite_g(Fl, F, Adj, 6, Pad); |
|
|
|
fwrite_g(Fl, F, Adj, P, Pad) when P >= 1 -> |
|
|
|
A = abs(Fl), |
|
|
|
E = if A < 1.0e-1 -> -2; |
|
|
|
A < 1.0e0 -> -1; |
|
|
|
A < 1.0e1 -> 0; |
|
|
|
A < 1.0e2 -> 1; |
|
|
|
A < 1.0e3 -> 2; |
|
|
|
A < 1.0e4 -> 3; |
|
|
|
true -> fwrite_f |
|
|
|
end, |
|
|
|
if P =< 1, E =:= -1; |
|
|
|
P - 1 > E, E >= -1 -> |
|
|
|
fwrite_f(Fl, F, Adj, P - 1 - E, Pad); |
|
|
|
P =< 1 -> |
|
|
|
fwrite_e(Fl, F, Adj, 2, Pad); |
|
|
|
true -> |
|
|
|
fwrite_e(Fl, F, Adj, P, Pad) |
|
|
|
end. |
|
|
|
|
|
|
|
|
|
|
|
%% string(String, Field, Adjust, Precision, PadChar) |
|
|
|
|
|
|
|
string(S, none, _Adj, none, _Pad) -> S; |
|
|
|
string(S, F, Adj, none, Pad) -> |
|
|
|
string_field(S, F, Adj, lists:flatlength(S), Pad); |
|
|
|
string(S, none, _Adj, P, Pad) -> |
|
|
|
string_field(S, P, left, lists:flatlength(S), Pad); |
|
|
|
string(S, F, Adj, P, Pad) when F >= P -> |
|
|
|
N = lists:flatlength(S), |
|
|
|
if F > P -> |
|
|
|
if N > P -> |
|
|
|
adjust(flat_trunc(S, P), chars(Pad, F - P), Adj); |
|
|
|
N < P -> |
|
|
|
adjust([S | chars(Pad, P - N)], chars(Pad, F - P), Adj); |
|
|
|
true -> % N == P |
|
|
|
adjust(S, chars(Pad, F - P), Adj) |
|
|
|
end; |
|
|
|
true -> % F == P |
|
|
|
string_field(S, F, Adj, N, Pad) |
|
|
|
end. |
|
|
|
|
|
|
|
string_field(S, F, _Adj, N, _Pad) when N > F -> |
|
|
|
flat_trunc(S, F); |
|
|
|
string_field(S, F, Adj, N, Pad) when N < F -> |
|
|
|
adjust(S, chars(Pad, F - N), Adj); |
|
|
|
string_field(S, _, _, _, _) -> % N == F |
|
|
|
S. |
|
|
|
|
|
|
|
%% unprefixed_integer(Int, Field, Adjust, Base, PadChar, Lowercase) |
|
|
|
%% -> [Char]. |
|
|
|
|
|
|
|
unprefixed_integer(Int, F, Adj, Base, Pad, Lowercase) |
|
|
|
when Base >= 2, Base =< 1 + $Z - $A + 10 -> |
|
|
|
if Int < 0 -> |
|
|
|
S = cond_lowercase(erlang:integer_to_list(-Int, Base), Lowercase), |
|
|
|
term([$- | S], F, Adj, none, Pad); |
|
|
|
true -> |
|
|
|
S = cond_lowercase(erlang:integer_to_list(Int, Base), Lowercase), |
|
|
|
term(S, F, Adj, none, Pad) |
|
|
|
end. |
|
|
|
|
|
|
|
%% prefixed_integer(Int, Field, Adjust, Base, PadChar, Prefix, Lowercase) |
|
|
|
%% -> [Char]. |
|
|
|
|
|
|
|
prefixed_integer(Int, F, Adj, Base, Pad, Prefix, Lowercase) |
|
|
|
when Base >= 2, Base =< 1 + $Z - $A + 10 -> |
|
|
|
if Int < 0 -> |
|
|
|
S = cond_lowercase(erlang:integer_to_list(-Int, Base), Lowercase), |
|
|
|
term([$-, Prefix | S], F, Adj, none, Pad); |
|
|
|
true -> |
|
|
|
S = cond_lowercase(erlang:integer_to_list(Int, Base), Lowercase), |
|
|
|
term([Prefix | S], F, Adj, none, Pad) |
|
|
|
end. |
|
|
|
|
|
|
|
%% char(Char, Field, Adjust, Precision, PadChar) -> [Char]. |
|
|
|
|
|
|
|
char(C, none, _Adj, none, _Pad) -> [C]; |
|
|
|
char(C, F, _Adj, none, _Pad) -> chars(C, F); |
|
|
|
char(C, none, _Adj, P, _Pad) -> chars(C, P); |
|
|
|
char(C, F, Adj, P, Pad) when F >= P -> |
|
|
|
adjust(chars(C, P), chars(Pad, F - P), Adj). |
|
|
|
|
|
|
|
%% newline(Field, Adjust, Precision, PadChar) -> [Char]. |
|
|
|
|
|
|
|
newline(none, _Adj, _P, _Pad) -> "\n"; |
|
|
|
newline(F, right, _P, _Pad) -> chars($\n, F). |
|
|
|
|
|
|
|
%% |
|
|
|
%% Utilities |
|
|
|
%% |
|
|
|
|
|
|
|
adjust(Data, [], _) -> Data; |
|
|
|
adjust(Data, Pad, left) -> [Data | Pad]; |
|
|
|
adjust(Data, Pad, right) -> [Pad | Data]. |
|
|
|
|
|
|
|
%% Flatten and truncate a deep list to at most N elements. |
|
|
|
flat_trunc(List, N) when is_integer(N), N >= 0 -> |
|
|
|
flat_trunc(List, N, []). |
|
|
|
|
|
|
|
flat_trunc(L, 0, R) when is_list(L) -> |
|
|
|
lists:reverse(R); |
|
|
|
flat_trunc([H | T], N, R) -> |
|
|
|
flat_trunc(T, N - 1, [H | R]); |
|
|
|
flat_trunc([], _, R) -> |
|
|
|
lists:reverse(R). |
|
|
|
|
|
|
|
%% A deep version of string:chars/2,3 |
|
|
|
|
|
|
|
chars(_C, 0) -> |
|
|
|
[]; |
|
|
|
chars(C, 1) -> |
|
|
|
[C]; |
|
|
|
chars(C, 2) -> |
|
|
|
[C, C]; |
|
|
|
chars(C, 3) -> |
|
|
|
[C, C, C]; |
|
|
|
chars(C, N) when is_integer(N), (N band 1) =:= 0 -> |
|
|
|
S = chars(C, N bsr 1), |
|
|
|
[S | S]; |
|
|
|
chars(C, N) when is_integer(N) -> |
|
|
|
S = chars(C, N bsr 1), |
|
|
|
[C, S | S]. |
|
|
|
|
|
|
|
%chars(C, N, Tail) -> |
|
|
|
% [chars(C, N)|Tail]. |
|
|
|
|
|
|
|
%% Lowercase conversion |
|
|
|
|
|
|
|
cond_lowercase(String, true) -> |
|
|
|
lowercase(String); |
|
|
|
cond_lowercase(String, false) -> |
|
|
|
String. |
|
|
|
|
|
|
|
lowercase([H | T]) when is_integer(H), H >= $A, H =< $Z -> |
|
|
|
[(H - $A + $a) | lowercase(T)]; |
|
|
|
lowercase([H | T]) -> |
|
|
|
[H | lowercase(T)]; |
|
|
|
lowercase([]) -> |
|
|
|
[]. |